dte test coverage


Directory: ./
Coverage: low: ≥ 0% medium: ≥ 50.0% high: ≥ 85.0%
Coverage Exec / Excl / Total
Lines: 98.7% 1089 / 4 / 1107
Functions: 100.0% 33 / 0 / 33
Branches: 88.7% 55 / 2 / 64

test/terminal.c
Line Branch Exec Source
1 #include "test.h"
2 #include "terminal/color.h"
3 #include "terminal/cursor.h"
4 #include "terminal/feature.h"
5 #include "terminal/input.h"
6 #include "terminal/key.h"
7 #include "terminal/linux.h"
8 #include "terminal/osc52.h"
9 #include "terminal/output.h"
10 #include "terminal/parse.h"
11 #include "terminal/rxvt.h"
12 #include "terminal/style.h"
13 #include "terminal/terminal.h"
14 #include "ui.h" // update_term_title()
15 #include "util/bit.h"
16 #include "util/str-array.h"
17 #include "util/unicode.h"
18 #include "util/utf8.h"
19 #include "util/xsnprintf.h"
20
21 #define TFLAG(flags) (KEYCODE_QUERY_REPLY_BIT | (flags))
22 #define IEXPECT_KEYCODE_EQ(a, b, seq, seq_len) IEXPECT(keycode_eq, a, b, seq, seq_len)
23 #define EXPECT_PARSE_SEQ(seq, expkey) EXPECT(parse_seq, seq, STRLEN(seq), expkey)
24 #define EXPECT_PARSE_SEQN(seq, explen, expkey) EXPECT(parse_seq, seq, explen, expkey)
25
26 1637 static void iexpect_keycode_eq (
27 TestContext *ctx,
28 const char *file,
29 int line,
30 size_t idx,
31 KeyCode a,
32 KeyCode b,
33 const char *seq,
34 size_t seq_len
35 ) {
36
1/2
✓ Branch 2 → 3 taken 1637 times.
✗ Branch 2 → 5 not taken.
1637 if (likely(a == b)) {
37 1637 test_pass(ctx);
38 1637 return;
39 }
40
41 StringView seq_sv = string_view(seq, seq_len);
42 char keystr_a[KEYCODE_STR_BUFSIZE];
43 char keystr_b[KEYCODE_STR_BUFSIZE];
44 char printable_seq[64];
45 keycode_to_str(a, keystr_a);
46 keycode_to_str(b, keystr_b);
47 u_make_printable(seq_sv, printable_seq, sizeof(printable_seq), 0);
48
49 test_fail(
50 ctx, file, line,
51 "Test #%zu: key codes not equal: 0x%02x, 0x%02x (%s, %s); input: %s",
52 ++idx, a, b, keystr_a, keystr_b, printable_seq
53 );
54 }
55
56 291 static void expect_parse_seq (
57 TestContext *ctx,
58 const char *file,
59 int line,
60 const char *seq,
61 ssize_t expected_length,
62 KeyCode expected_key
63 ) {
64 291 KeyCode key = 0x18;
65 291 ssize_t parsed_length = term_parse_sequence(seq, strlen(seq), &key);
66
1/2
✗ Branch 3 → 4 not taken.
✓ Branch 3 → 5 taken 291 times.
291 if (unlikely(parsed_length != expected_length)) {
67 test_fail (
68 ctx, file, line,
69 "term_parse_sequence() returned %zd; expected %zd",
70 parsed_length, expected_length
71 );
72 14 return;
73 }
74
75 291 static_assert(TPARSE_PARTIAL_MATCH == -1);
76 291 static_assert(TPARSE_NO_MATCH == 0);
77
2/2
✓ Branch 5 → 6 taken 14 times.
✓ Branch 5 → 8 taken 277 times.
291 if (expected_length <= TPARSE_NO_MATCH) {
78 14 test_pass(ctx);
79 14 return;
80 }
81
82
1/2
✗ Branch 8 → 9 not taken.
✓ Branch 8 → 13 taken 277 times.
277 if (unlikely(key != expected_key)) {
83 char str[2][KEYCODE_STR_BUFSIZE];
84 keycode_to_str(key, str[0]);
85 keycode_to_str(expected_key, str[1]);
86 test_fail (
87 ctx, file, line,
88 "Key codes not equal: 0x%02x, 0x%02x (%s, %s)",
89 key, expected_key, str[0], str[1]
90 );
91 return;
92 }
93
94
2/2
✓ Branch 19 → 14 taken 1801 times.
✓ Branch 19 → 20 taken 277 times.
2078 for (size_t n = expected_length - 1; n != 0; n--) {
95 1801 parsed_length = term_parse_sequence(seq, n, &key);
96
1/2
✓ Branch 15 → 16 taken 1801 times.
✗ Branch 15 → 17 not taken.
1801 if (likely(parsed_length == TPARSE_PARTIAL_MATCH)) {
97 1801 continue;
98 }
99
100 test_fail (
101 ctx, file, line,
102 "Parsing truncated sequence of %zu bytes returned %zd; expected %zd",
103 n, parsed_length, (ssize_t)TPARSE_PARTIAL_MATCH
104 );
105
106 return;
107 }
108
109 277 test_pass(ctx);
110 }
111
112 1 static void test_parse_rgb(TestContext *ctx)
113 {
114 1 EXPECT_EQ(parse_rgb(STRN("f01cff")), COLOR_RGB(0xf01cff));
115 1 EXPECT_EQ(parse_rgb(STRN("011011")), COLOR_RGB(0x011011));
116 1 EXPECT_EQ(parse_rgb(STRN("12aC90")), COLOR_RGB(0x12ac90));
117 1 EXPECT_EQ(parse_rgb(STRN("fffffF")), COLOR_RGB(0xffffff));
118 1 EXPECT_EQ(parse_rgb(STRN("fa0")), COLOR_RGB(0xffaa00));
119 1 EXPECT_EQ(parse_rgb(STRN("123")), COLOR_RGB(0x112233));
120 1 EXPECT_EQ(parse_rgb(STRN("fff")), COLOR_RGB(0xffffff));
121 1 EXPECT_EQ(parse_rgb(STRN("ABC")), COLOR_RGB(0xaabbcc));
122 1 EXPECT_EQ(parse_rgb(STRN("0F0")), COLOR_RGB(0x00ff00));
123 1 EXPECT_EQ(parse_rgb(STRN("000")), COLOR_RGB(0x000000));
124 1 EXPECT_EQ(parse_rgb(STRN("fffffg")), COLOR_INVALID);
125 1 EXPECT_EQ(parse_rgb(STRN(".")), COLOR_INVALID);
126 1 EXPECT_EQ(parse_rgb(STRN("")), COLOR_INVALID);
127 1 EXPECT_EQ(parse_rgb(STRN("11223")), COLOR_INVALID);
128 1 EXPECT_EQ(parse_rgb(STRN("efg")), COLOR_INVALID);
129 1 EXPECT_EQ(parse_rgb(STRN("12")), COLOR_INVALID);
130 1 EXPECT_EQ(parse_rgb(STRN("ffff")), COLOR_INVALID);
131 1 EXPECT_EQ(parse_rgb(STRN("00")), COLOR_INVALID);
132 1 EXPECT_EQ(parse_rgb(STRN("1234567")), COLOR_INVALID);
133 1 EXPECT_EQ(parse_rgb(STRN("123456789")), COLOR_INVALID);
134 1 }
135
136 1 static void test_parse_term_style(TestContext *ctx)
137 {
138 1 static const struct {
139 const char *const strs[4];
140 TermStyle expected_style;
141 } tests[] = {
142 {{"bold", "red", "yellow"}, {.fg = COLOR_RED, .bg = COLOR_YELLOW, .attr = ATTR_BOLD}},
143 {{"#ff0000"}, {.fg = COLOR_RGB(0xff0000), .bg = -1}},
144 {{"#f00a9c", "reverse"}, {.fg = COLOR_RGB(0xf00a9c), .bg = -1, .attr = ATTR_REVERSE}},
145 {{"black", "#00ffff"}, {.fg = COLOR_BLACK, .bg = COLOR_RGB(0x00ffff)}},
146 {{"#123456", "#abcdef"}, {.fg = COLOR_RGB(0x123456), .bg = COLOR_RGB(0xabcdef)}},
147 {{"#123", "#fa0"}, {.fg = COLOR_RGB(0x112233), .bg = COLOR_RGB(0xffaa00)}},
148 {{"#A1B2C3", "gray"}, {.fg = COLOR_RGB(0xa1b2c3), .bg = COLOR_GRAY}},
149 {{"red", "strikethrough"}, {.fg = COLOR_RED, .bg = -1, .attr = ATTR_STRIKETHROUGH}},
150 {{"5/5/5"}, {.fg = 231, .bg = COLOR_DEFAULT}},
151 {{"1/3/0", "0/5/2", "italic"}, {.fg = 70, .bg = 48, .attr = ATTR_ITALIC}},
152 {{"-1", "-2"}, {.fg = COLOR_DEFAULT, .bg = COLOR_KEEP}},
153 {{"keep", "red", "keep"}, {.fg = -2, .bg = COLOR_RED, .attr = ATTR_KEEP}},
154 {{"bold", "blink"}, {.fg = -1, .bg = -1, .attr = ATTR_BOLD | ATTR_BLINK}},
155 {{"0", "255", "underline"}, {.fg = COLOR_BLACK, .bg = 255, .attr = ATTR_UNDERLINE}},
156 {{"white", "green", "dim"}, {.fg = COLOR_WHITE, .bg = COLOR_GREEN, .attr = ATTR_DIM}},
157 {{"white", "green", "lowintensity"}, {.fg = COLOR_WHITE, .bg = COLOR_GREEN, .attr = ATTR_DIM}},
158 {{"lightred", "lightyellow"}, {.fg = COLOR_LIGHTRED, .bg = COLOR_LIGHTYELLOW}},
159 {{"darkgray", "lightgreen"}, {.fg = COLOR_DARKGRAY, .bg = COLOR_LIGHTGREEN}},
160 {{"lightblue", "lightcyan"}, {.fg = COLOR_LIGHTBLUE, .bg = COLOR_LIGHTCYAN}},
161 {{"lightmagenta"}, {.fg = COLOR_LIGHTMAGENTA, .bg = COLOR_DEFAULT}},
162 {{"keep", "254", "keep"}, {.fg = COLOR_KEEP, .bg = 254, .attr = ATTR_KEEP}},
163 {{"keep", "keep"}, {.fg = COLOR_KEEP, .bg = COLOR_KEEP}},
164 {{"red", "green", "keep"}, {.fg = COLOR_RED, .bg = COLOR_GREEN, .attr = ATTR_KEEP}},
165 {{"1", "2", "invisible"}, {.fg = COLOR_RED, .bg = COLOR_GREEN, .attr = ATTR_INVIS}},
166 {{"default", "0/0/0"}, {.fg = COLOR_DEFAULT, .bg = 16}},
167 {{"2/3/4", "5/5/5"}, {.fg = 110, .bg = 231}},
168 {{NULL}, {.fg = COLOR_DEFAULT, .bg = COLOR_DEFAULT}},
169 };
170
171 1 static const struct {
172 ssize_t expected_return;
173 const char *const strs[4];
174 } invalid_tests[] = {
175 {2, {"red", "blue", "invalid"}},
176 {-1, {"cyan", "magenta", "yellow"}},
177 {0, {"invalid", "default", "bold"}},
178 {1, {"italic", "invalid"}},
179 {2, {"red", "blue", ""}},
180 {2, {"24", "#abc", "dims"}},
181 {0, {""}},
182 {0, {"."}},
183 {0, {"#"}},
184 {0, {"-3"}},
185 {0, {"256"}},
186 {0, {"brightred"}},
187 {0, {"lighttblack"}},
188 {0, {"lightwhite"}},
189 {0, {"#fffffg"}},
190 {0, {"#12345"}},
191 {0, {"123456"}},
192 {0, {"//0/0"}},
193 {0, {"0/0/:"}},
194 {0, {"light_"}},
195 {0, {"\xFF/0/0"}},
196 {0, {"1/2/\x9E"}},
197 };
198
199
2/2
✓ Branch 10 → 3 taken 27 times.
✓ Branch 10 → 19 taken 1 time.
28 FOR_EACH_I(i, tests) {
200 27 const TermStyle expected = tests[i].expected_style;
201 27 TermStyle parsed = {.fg = COLOR_INVALID, .bg = COLOR_INVALID};
202 27 char **strs = (char**)tests[i].strs;
203 27 size_t nstrs = string_array_length(strs);
204 27 ssize_t ret = parse_term_style(&parsed, strs, nstrs);
205 27 IEXPECT_EQ(ret, nstrs);
206 27 IEXPECT_EQ(parsed.fg, expected.fg);
207 27 IEXPECT_EQ(parsed.bg, expected.bg);
208 27 IEXPECT_EQ(parsed.attr, expected.attr);
209 27 IEXPECT_TRUE(same_style(&parsed, &expected));
210 }
211
212
2/2
✓ Branch 19 → 11 taken 22 times.
✓ Branch 19 → 20 taken 1 time.
23 FOR_EACH_I(i, invalid_tests) {
213 22 const TermStyle expected = {.fg = COLOR_INVALID, .bg = COLOR_INVALID};
214 22 TermStyle parsed = expected;
215 22 char **strs = (char**)invalid_tests[i].strs;
216 22 size_t nstrs = string_array_length(strs);
217 22 ssize_t ret = parse_term_style(&parsed, strs, nstrs);
218 22 IEXPECT_EQ(ret, invalid_tests[i].expected_return);
219 22 IEXPECT_NE(ret, nstrs);
220 22 IEXPECT_EQ(parsed.fg, COLOR_INVALID);
221 22 IEXPECT_EQ(parsed.bg, COLOR_INVALID);
222 22 IEXPECT_EQ(parsed.attr, 0);
223 22 IEXPECT_TRUE(same_style(&parsed, &expected));
224 }
225 1 }
226
227 1 static void test_color_to_nearest(TestContext *ctx)
228 {
229 1 static const struct {
230 int32_t input;
231 int32_t expected_rgb;
232 int32_t expected_256;
233 int32_t expected_16;
234 } tests[] = {
235 // ECMA-48 colors
236 {0, 0, 0, 0},
237 {5, 5, 5, 5},
238 {7, 7, 7, 7},
239
240 // aixterm-style colors
241 {8, 8, 8, 8},
242 {10, 10, 10, 10},
243 {15, 15, 15, 15},
244
245 // xterm 256 palette colors
246 {25, 25, 25, COLOR_BLUE},
247 {87, 87, 87, COLOR_LIGHTCYAN},
248 {88, 88, 88, COLOR_RED},
249 {90, 90, 90, COLOR_MAGENTA},
250 {96, 96, 96, COLOR_MAGENTA},
251 {178, 178, 178, COLOR_YELLOW},
252 {179, 179, 179, COLOR_YELLOW},
253
254 // RGB colors with exact xterm 6x6x6 cube equivalents
255 {COLOR_RGB(0x000000), 16, 16, COLOR_BLACK},
256 {COLOR_RGB(0x000087), 18, 18, COLOR_BLUE},
257 {COLOR_RGB(0x0000FF), 21, 21, COLOR_LIGHTBLUE},
258 {COLOR_RGB(0x00AF87), 36, 36, COLOR_GREEN},
259 {COLOR_RGB(0x00FF00), 46, 46, COLOR_LIGHTGREEN},
260 {COLOR_RGB(0x870000), 88, 88, COLOR_RED},
261 {COLOR_RGB(0xFF0000), 196, 196, COLOR_LIGHTRED},
262 {COLOR_RGB(0xFFD700), 220, 220, COLOR_YELLOW},
263 {COLOR_RGB(0xFFFF5F), 227, 227, COLOR_LIGHTYELLOW},
264 {COLOR_RGB(0xFFFFFF), 231, 231, COLOR_WHITE},
265
266 // RGB colors with exact xterm grayscale equivalents
267 {COLOR_RGB(0x080808), 232, 232, COLOR_BLACK},
268 {COLOR_RGB(0x121212), 233, 233, COLOR_BLACK},
269 {COLOR_RGB(0x6C6C6C), 242, 242, COLOR_DARKGRAY},
270 {COLOR_RGB(0xA8A8A8), 248, 248, COLOR_GRAY},
271 {COLOR_RGB(0xB2B2B2), 249, 249, COLOR_GRAY},
272 {COLOR_RGB(0xBCBCBC), 250, 250, COLOR_WHITE},
273 {COLOR_RGB(0xEEEEEE), 255, 255, COLOR_WHITE},
274
275 // RGB colors with NO exact xterm equivalents
276 {COLOR_RGB(0x00FF88), COLOR_RGB(0x00FF88), 48, COLOR_LIGHTGREEN},
277 {COLOR_RGB(0xFF0001), COLOR_RGB(0xFF0001), 196, COLOR_LIGHTRED},
278 {COLOR_RGB(0xAABBCC), COLOR_RGB(0xAABBCC), 146, COLOR_LIGHTBLUE},
279 {COLOR_RGB(0x010101), COLOR_RGB(0x010101), 16, COLOR_BLACK},
280 {COLOR_RGB(0x070707), COLOR_RGB(0x070707), 232, COLOR_BLACK},
281 {COLOR_RGB(0x080809), COLOR_RGB(0x080809), 232, COLOR_BLACK},
282 {COLOR_RGB(0xBABABA), COLOR_RGB(0xBABABA), 250, COLOR_WHITE},
283 {COLOR_RGB(0xEEEEED), COLOR_RGB(0xEEEEED), 255, COLOR_WHITE},
284 {COLOR_RGB(0xEFEFEF), COLOR_RGB(0xEFEFEF), 255, COLOR_WHITE},
285 {COLOR_RGB(0xF0F0F0), COLOR_RGB(0xF0F0F0), 255, COLOR_WHITE},
286 {COLOR_RGB(0xF6F6F6), COLOR_RGB(0xF6F6F6), 255, COLOR_WHITE},
287 {COLOR_RGB(0xF7F7F7), COLOR_RGB(0xF7F7F7), 231, COLOR_WHITE},
288 {COLOR_RGB(0xFEFEFE), COLOR_RGB(0xFEFEFE), 231, COLOR_WHITE},
289 };
290
291
2/2
✓ Branch 16 → 3 taken 43 times.
✓ Branch 16 → 17 taken 1 time.
44 FOR_EACH_I(i, tests) {
292 43 const int32_t c = tests[i].input;
293 43 IEXPECT_EQ(color_to_nearest(c, TFLAG_TRUE_COLOR, false), c);
294 43 IEXPECT_EQ(color_to_nearest(c, TFLAG_TRUE_COLOR, true), tests[i].expected_rgb);
295 43 IEXPECT_EQ(color_to_nearest(c, TFLAG_256_COLOR, false), tests[i].expected_256);
296 43 IEXPECT_EQ(color_to_nearest(c, TFLAG_16_COLOR, false), tests[i].expected_16);
297 43 IEXPECT_EQ(color_to_nearest(c, TFLAG_8_COLOR, false), tests[i].expected_16 & 7);
298 43 IEXPECT_EQ(color_to_nearest(c, 0, false), COLOR_DEFAULT);
299 }
300
301 static const uint8_t color_stops[6] = {0x00, 0x5f, 0x87, 0xaf, 0xd7, 0xff};
302 size_t count = 0;
303
304
2/2
✓ Branch 40 → 18 taken 5 times.
✓ Branch 40 → 41 taken 1 time.
6 for (size_t i = 1; i < ARRAYLEN(color_stops); i++) {
305 5 uint8_t min = color_stops[i - 1];
306 5 uint8_t max = color_stops[i];
307 5 uint8_t mid = min + ((max - min) / 2);
308
2/2
✓ Branch 38 → 19 taken 260 times.
✓ Branch 38 → 39 taken 5 times.
265 for (unsigned int b = min; b <= max; b++, count++) {
309 260 int32_t orig = COLOR_RGB(b);
310 260 int32_t nearest = color_to_nearest(orig, TFLAG_256_COLOR, false);
311
2/2
✓ Branch 20 → 21 taken 127 times.
✓ Branch 20 → 22 taken 133 times.
260 size_t nearest_stop_idx = (b < mid) ? i - 1 : i;
312 260 EXPECT_TRUE(nearest >= 16);
313 260 EXPECT_TRUE(nearest <= 255);
314
2/2
✓ Branch 24 → 25 taken 220 times.
✓ Branch 24 → 26 taken 40 times.
260 if (nearest < 232) {
315 220 EXPECT_EQ(nearest, nearest_stop_idx + 16);
316 } else {
317 40 int32_t v = orig & ~COLOR_FLAG_RGB;
318 40 EXPECT_TRUE(v >= 0x0D);
319 40 EXPECT_TRUE(v <= 0x34);
320 40 EXPECT_EQ(nearest, 232 + (((b / 3) - 3) / 10));
321 }
322
2/4
✓ Branch 29 → 30 taken 260 times.
✗ Branch 29 → 31 not taken.
✗ Branch 30 → 31 not taken.
✓ Branch 30 → 34 taken 260 times.
260 if (nearest == min || nearest == max) {
323 EXPECT_EQ(nearest, b);
324 EXPECT_EQ(nearest, color_to_nearest(orig, TFLAG_TRUE_COLOR, true));
325 }
326 445 EXPECT_EQ(nearest_stop_idx, ((uint8_t)b - (b < 75 ? 7 : 35)) / 40);
327 }
328 }
329
330 1 EXPECT_EQ(count, 255 + ARRAYLEN(color_stops) - 1);
331 1 }
332
333 1 static void test_color_to_str(TestContext *ctx)
334 {
335 1 static const struct {
336 char expected_string[15];
337 uint8_t expected_len;
338 int32_t color;
339 } tests[] = {
340 {STRN("keep"), COLOR_KEEP},
341 {STRN("default"), COLOR_DEFAULT},
342 {STRN("black"), COLOR_BLACK},
343 {STRN("gray"), COLOR_GRAY},
344 {STRN("darkgray"), COLOR_DARKGRAY},
345 {STRN("lightmagenta"), COLOR_LIGHTMAGENTA},
346 {STRN("white"), COLOR_WHITE},
347 {STRN("16"), 16},
348 {STRN("255"), 255},
349 {STRN("#abcdef"), COLOR_RGB(0xabcdef)},
350 {STRN("#00001e"), COLOR_RGB(0x00001e)},
351 {STRN("#000000"), COLOR_RGB(0x000000)},
352 {STRN("#100001"), COLOR_RGB(0x100001)},
353 {STRN("#ffffff"), COLOR_RGB(0xffffff)},
354 };
355
356 1 char buf[COLOR_STR_BUFSIZE] = "";
357
2/2
✓ Branch 7 → 3 taken 14 times.
✓ Branch 7 → 8 taken 1 time.
15 FOR_EACH_I(i, tests) {
358 14 int32_t color = tests[i].color;
359 14 ASSERT_TRUE(color_is_valid(color));
360 14 size_t len = color_to_str(buf, color);
361 14 EXPECT_MEMEQ(buf, len, tests[i].expected_string, tests[i].expected_len);
362 }
363 1 }
364
365 1 static void test_term_style_to_string(TestContext *ctx)
366 {
367 1 static const struct {
368 const char *expected_string;
369 const TermStyle style;
370 } tests[] = {
371 {"red yellow bold", {.fg = COLOR_RED, .bg = COLOR_YELLOW, .attr = ATTR_BOLD}},
372 {"#ff0000", {.fg = COLOR_RGB(0xff0000), .bg = -1}},
373 {"#f00a9c reverse", {.fg = COLOR_RGB(0xf00a9c), .bg = -1, .attr = ATTR_REVERSE}},
374 {"black #00ffff", {.fg = COLOR_BLACK, .bg = COLOR_RGB(0x00ffff)}},
375 {"#010900", {.fg = COLOR_RGB(0x010900), .bg = COLOR_DEFAULT}},
376 {"red strikethrough", {.fg = COLOR_RED, .bg = -1, .attr = ATTR_STRIKETHROUGH}},
377 {"231", {.fg = 231, .bg = COLOR_DEFAULT}},
378 {"70 48 italic", {.fg = 70, .bg = 48, .attr = ATTR_ITALIC}},
379 {"default keep", {.fg = COLOR_DEFAULT, .bg = COLOR_KEEP}},
380 {"keep red keep", {.fg = COLOR_KEEP, .bg = COLOR_RED, .attr = ATTR_KEEP}},
381 {"88 16 blink bold", {.fg = 88, .bg = 16, .attr = ATTR_BOLD | ATTR_BLINK}},
382 {"black 255 underline", {.fg = COLOR_BLACK, .bg = 255, .attr = ATTR_UNDERLINE}},
383 {"white green dim", {.fg = COLOR_WHITE, .bg = COLOR_GREEN, .attr = ATTR_DIM}},
384 {"darkgray lightgreen", {.fg = COLOR_DARKGRAY, .bg = COLOR_LIGHTGREEN}},
385 {"lightmagenta", {.fg = COLOR_LIGHTMAGENTA, .bg = COLOR_DEFAULT}},
386 {"keep 254 keep", {.fg = COLOR_KEEP, .bg = 254, .attr = ATTR_KEEP}},
387 {"default", {.fg = COLOR_DEFAULT, .bg = COLOR_DEFAULT}},
388 {"default bold", {.fg = COLOR_DEFAULT, .bg = COLOR_DEFAULT, .attr = ATTR_BOLD}},
389 {"default default keep", {.fg = COLOR_DEFAULT, .bg = COLOR_DEFAULT, .attr = ATTR_KEEP}},
390 };
391
392 1 char buf[TERM_STYLE_BUFSIZE];
393
2/2
✓ Branch 6 → 3 taken 19 times.
✓ Branch 6 → 7 taken 1 time.
20 FOR_EACH_I(i, tests) {
394 19 const char *str = term_style_to_string(buf, &tests[i].style);
395 19 EXPECT_STREQ(str, tests[i].expected_string);
396 }
397
398 // Ensure longest possible color string doesn't overflow the
399 // static buffer in term_style_to_string()
400 1 const TermStyle style = {
401 .fg = COLOR_LIGHTMAGENTA,
402 .bg = COLOR_LIGHTMAGENTA,
403 .attr = ATTR_ALL,
404 };
405
406 1 static const char expected[] =
407 "lightmagenta lightmagenta "
408 "keep underline reverse blink dim "
409 "bold invisible italic strikethrough"
410 ;
411
412 1 static_assert(sizeof(expected) - 1 == 94);
413 1 const char *str = term_style_to_string(buf, &style);
414 1 EXPECT_STREQ(str, expected);
415 1 }
416
417 1 static void test_cursor_mode_from_str(TestContext *ctx)
418 {
419 1 EXPECT_EQ(cursor_mode_from_str("default"), CURSOR_MODE_DEFAULT);
420 1 EXPECT_EQ(cursor_mode_from_str("insert"), CURSOR_MODE_INSERT);
421 1 EXPECT_EQ(cursor_mode_from_str("overwrite"), CURSOR_MODE_OVERWRITE);
422 1 EXPECT_EQ(cursor_mode_from_str("cmdline"), CURSOR_MODE_CMDLINE);
423 1 EXPECT_EQ(cursor_mode_from_str(""), NR_CURSOR_MODES);
424 1 EXPECT_EQ(cursor_mode_from_str("d"), NR_CURSOR_MODES);
425 1 EXPECT_EQ(cursor_mode_from_str("defaults"), NR_CURSOR_MODES);
426 1 EXPECT_EQ(cursor_mode_from_str("command"), NR_CURSOR_MODES);
427 1 }
428
429 1 static void test_cursor_type_from_str(TestContext *ctx)
430 {
431 1 EXPECT_EQ(cursor_type_from_str("default"), CURSOR_DEFAULT);
432 1 EXPECT_EQ(cursor_type_from_str("keep"), CURSOR_KEEP);
433 1 EXPECT_EQ(cursor_type_from_str("block"), CURSOR_STEADY_BLOCK);
434 1 EXPECT_EQ(cursor_type_from_str("bar"), CURSOR_STEADY_BAR);
435 1 EXPECT_EQ(cursor_type_from_str("underline"), CURSOR_STEADY_UNDERLINE);
436 1 EXPECT_EQ(cursor_type_from_str("blinking-block"), CURSOR_BLINKING_BLOCK);
437 1 EXPECT_EQ(cursor_type_from_str("blinking-bar"), CURSOR_BLINKING_BAR);
438 1 EXPECT_EQ(cursor_type_from_str("blinking-underline"), CURSOR_BLINKING_UNDERLINE);
439 1 EXPECT_EQ(cursor_type_from_str(""), CURSOR_INVALID);
440 1 EXPECT_EQ(cursor_type_from_str("b"), CURSOR_INVALID);
441 1 EXPECT_EQ(cursor_type_from_str("bars"), CURSOR_INVALID);
442 1 EXPECT_EQ(cursor_type_from_str("blinking-default"), CURSOR_INVALID);
443 1 EXPECT_EQ(cursor_type_from_str("blinking-keep"), CURSOR_INVALID);
444 1 }
445
446 1 static void test_cursor_color_from_str(TestContext *ctx)
447 {
448 1 EXPECT_EQ(cursor_color_from_str("keep"), COLOR_KEEP);
449 1 EXPECT_EQ(cursor_color_from_str("default"), COLOR_DEFAULT);
450 1 EXPECT_EQ(cursor_color_from_str("#f9a"), COLOR_RGB(0xFF99AA));
451 1 EXPECT_EQ(cursor_color_from_str("#123456"), COLOR_RGB(0x123456));
452 1 EXPECT_EQ(cursor_color_from_str("red"), COLOR_INVALID);
453 1 EXPECT_EQ(cursor_color_from_str(""), COLOR_INVALID);
454 1 EXPECT_EQ(cursor_color_from_str("#"), COLOR_INVALID);
455 1 EXPECT_EQ(cursor_color_from_str("0"), COLOR_INVALID);
456 1 EXPECT_EQ(cursor_color_from_str("#12345"), COLOR_INVALID);
457 1 EXPECT_EQ(cursor_color_from_str("#1234567"), COLOR_INVALID);
458 1 EXPECT_EQ(cursor_color_from_str("123456"), COLOR_INVALID);
459 1 }
460
461 1 static void test_cursor_color_to_str(TestContext *ctx)
462 {
463 1 char buf[COLOR_STR_BUFSIZE];
464 1 EXPECT_STREQ(cursor_color_to_str(buf, COLOR_DEFAULT), "default");
465 1 EXPECT_STREQ(cursor_color_to_str(buf, COLOR_KEEP), "keep");
466 1 EXPECT_STREQ(cursor_color_to_str(buf, COLOR_RGB(0x190AFE)), "#190afe");
467 1 }
468
469 1 static void test_same_cursor(TestContext *ctx)
470 {
471 1 TermCursorStyle a = get_default_cursor_style(CURSOR_MODE_DEFAULT);
472 1 EXPECT_EQ(a.type, CURSOR_DEFAULT);
473 1 EXPECT_EQ(a.color, COLOR_DEFAULT);
474 1 EXPECT_TRUE(same_cursor(&a, &a));
475
476 1 TermCursorStyle b = get_default_cursor_style(CURSOR_MODE_INSERT);
477 1 EXPECT_EQ(b.type, CURSOR_KEEP);
478 1 EXPECT_EQ(b.color, COLOR_KEEP);
479 1 EXPECT_TRUE(same_cursor(&b, &b));
480
481 1 EXPECT_FALSE(same_cursor(&a, &b));
482 1 b.type = CURSOR_DEFAULT;
483 1 EXPECT_FALSE(same_cursor(&a, &b));
484 1 b.color = COLOR_DEFAULT;
485 1 EXPECT_TRUE(same_cursor(&a, &b));
486 1 }
487
488 1 static void test_mask_style(TestContext *ctx)
489 {
490 1 const TermStyle keep_all = term_style(COLOR_KEEP, COLOR_KEEP, ATTR_KEEP);
491 1 const TermStyle keep_fg = term_style(COLOR_KEEP, 93, ATTR_STRIKETHROUGH);
492 1 const TermStyle keep_bg = term_style(241, COLOR_KEEP, ATTR_REVERSE);
493 1 const TermStyle a = term_style(COLOR_BLUE, COLOR_CYAN, ATTR_BOLD);
494 1 const TermStyle b = term_style(100, 105, ATTR_DIM);
495
496 1 TermStyle style = a;
497 1 mask_style(&style, &keep_all);
498 1 EXPECT_EQ(style.fg, a.fg);
499 1 EXPECT_EQ(style.bg, a.bg);
500 1 EXPECT_EQ(style.attr, a.attr);
501 1 EXPECT_TRUE(same_style(&style, &a));
502
503 1 style = a;
504 1 mask_style(&style, &keep_bg);
505 1 EXPECT_EQ(style.fg, keep_bg.fg);
506 1 EXPECT_EQ(style.bg, a.bg);
507 1 EXPECT_EQ(style.attr, keep_bg.attr);
508
509 1 style = a;
510 1 mask_style(&style, &keep_fg);
511 1 EXPECT_EQ(style.fg, a.fg);
512 1 EXPECT_EQ(style.bg, keep_fg.bg);
513 1 EXPECT_EQ(style.attr, keep_fg.attr);
514
515 1 style = a;
516 1 mask_style2(&style, &b, false);
517 1 EXPECT_EQ(style.bg, a.bg);
518
519 1 style = a;
520 1 mask_style2(&style, &b, true);
521 1 EXPECT_EQ(style.bg, b.bg);
522
523 1 style = a;
524 1 mask_style2(&style, &keep_bg, true);
525 1 EXPECT_EQ(style.bg, a.bg);
526
527 1 style = a;
528 1 mask_style2(&style, &keep_bg, false);
529 1 EXPECT_EQ(style.bg, a.bg);
530 1 }
531
532 1 static void test_term_parse_csi_params(TestContext *ctx)
533 {
534 1 TermControlParams csi = {.nparams = 0};
535 1 StringView s = strview("\033[901;0;55mx");
536 1 size_t n = term_parse_csi_params(s.data, s.length, 2, &csi);
537 1 EXPECT_EQ(n, s.length - 1);
538 1 EXPECT_EQ(csi.nparams, 3);
539 1 EXPECT_EQ(csi.params[0][0], 901);
540 1 EXPECT_EQ(csi.params[1][0], 0);
541 1 EXPECT_EQ(csi.params[2][0], 55);
542 1 EXPECT_EQ(csi.nr_intermediate, 0);
543 1 EXPECT_EQ(csi.final_byte, 'm');
544 1 EXPECT_FALSE(csi.have_subparams);
545 1 EXPECT_FALSE(csi.unhandled_bytes);
546
547 1 csi = (TermControlParams){.nparams = 0};
548 1 s = strview("\033[123;09;56:78:99m");
549 1 n = term_parse_csi_params(s.data, s.length, 2, &csi);
550 1 EXPECT_EQ(n, s.length);
551 1 EXPECT_EQ(csi.nparams, 3);
552 1 static_assert(ARRAYLEN(csi.nsub) >= 4);
553 1 static_assert(ARRAYLEN(csi.nsub) == ARRAYLEN(csi.params));
554 1 EXPECT_EQ(csi.nsub[0], 1);
555 1 EXPECT_EQ(csi.nsub[1], 1);
556 1 EXPECT_EQ(csi.nsub[2], 3);
557 1 EXPECT_EQ(csi.nsub[3], 0);
558 1 EXPECT_EQ(csi.params[0][0], 123);
559 1 EXPECT_EQ(csi.params[1][0], 9);
560 1 EXPECT_EQ(csi.params[2][0], 56);
561 1 EXPECT_EQ(csi.params[2][1], 78);
562 1 EXPECT_EQ(csi.params[2][2], 99);
563 1 EXPECT_EQ(csi.params[3][0], 0);
564 1 EXPECT_EQ(csi.nr_intermediate, 0);
565 1 EXPECT_EQ(csi.final_byte, 'm');
566 1 EXPECT_TRUE(csi.have_subparams);
567 1 EXPECT_FALSE(csi.unhandled_bytes);
568
569 1 csi = (TermControlParams){.nparams = 0};
570 1 s = strview("\033[1:2:3;44:55;;6~");
571 1 n = term_parse_csi_params(s.data, s.length, 2, &csi);
572 1 EXPECT_EQ(n, s.length);
573 1 EXPECT_EQ(csi.nparams, 4);
574 1 EXPECT_EQ(csi.nsub[0], 3);
575 1 EXPECT_EQ(csi.nsub[1], 2);
576 1 EXPECT_EQ(csi.nsub[2], 1);
577 1 EXPECT_EQ(csi.params[0][0], 1);
578 1 EXPECT_EQ(csi.params[0][1], 2);
579 1 EXPECT_EQ(csi.params[0][2], 3);
580 1 EXPECT_EQ(csi.params[1][0], 44);
581 1 EXPECT_EQ(csi.params[1][1], 55);
582 1 EXPECT_EQ(csi.params[2][0], 0);
583 1 EXPECT_EQ(csi.params[3][0], 6);
584 1 EXPECT_EQ(csi.nr_intermediate, 0);
585 1 EXPECT_EQ(csi.final_byte, '~');
586 1 EXPECT_TRUE(csi.have_subparams);
587 1 EXPECT_FALSE(csi.unhandled_bytes);
588
589 1 csi = (TermControlParams){.nparams = 0};
590 1 s = strview("\033[+2p");
591 1 n = term_parse_csi_params(s.data, s.length, 2, &csi);
592 1 EXPECT_EQ(n, s.length);
593 1 EXPECT_EQ(csi.nparams, 1);
594 1 EXPECT_EQ(csi.nsub[0], 1);
595 1 EXPECT_EQ(csi.params[0][0], 2);
596 1 EXPECT_EQ(csi.nr_intermediate, 1);
597 1 EXPECT_EQ(csi.intermediate[0], '+');
598 1 EXPECT_EQ(csi.final_byte, 'p');
599 1 EXPECT_FALSE(csi.have_subparams);
600 1 EXPECT_FALSE(csi.unhandled_bytes);
601
602 1 csi = (TermControlParams){.nparams = 0};
603 1 s = strview("\033[?47;1$y");
604 1 n = term_parse_csi_params(s.data, s.length, 2, &csi);
605 1 EXPECT_EQ(n, s.length);
606 1 EXPECT_EQ(csi.nparams, 2);
607 1 EXPECT_EQ(csi.params[0][0], 47);
608 1 EXPECT_EQ(csi.params[1][0], 1);
609 1 EXPECT_EQ(csi.nr_intermediate, 1);
610 1 EXPECT_EQ(csi.intermediate[0], '$');
611 1 EXPECT_FALSE(csi.have_subparams);
612 // Note: question mark param prefixes are handled in parse_csi() instead
613 // of term_parse_csi_params(), so the latter reports it as an unhandled byte
614 1 EXPECT_TRUE(csi.unhandled_bytes);
615 1 }
616
617 1 static void test_term_parse_sequence(TestContext *ctx)
618 {
619 1 EXPECT_PARSE_SEQN("", TPARSE_PARTIAL_MATCH, 0);
620 1 EXPECT_PARSE_SEQN("x", TPARSE_NO_MATCH, 0);
621 1 EXPECT_PARSE_SEQN("\033q", TPARSE_NO_MATCH, 0);
622 1 EXPECT_PARSE_SEQ("\033\\", KEY_IGNORE);
623
624 1 EXPECT_PARSE_SEQ("\033[Z", MOD_SHIFT | KEY_TAB);
625 1 EXPECT_PARSE_SEQ("\033[1;2A", MOD_SHIFT | KEY_UP);
626 1 EXPECT_PARSE_SEQ("\033[1;2B", MOD_SHIFT | KEY_DOWN);
627 1 EXPECT_PARSE_SEQ("\033[1;2C", MOD_SHIFT | KEY_RIGHT);
628 1 EXPECT_PARSE_SEQ("\033[1;2D", MOD_SHIFT | KEY_LEFT);
629 1 EXPECT_PARSE_SEQ("\033[1;2E", MOD_SHIFT | KEY_BEGIN);
630 1 EXPECT_PARSE_SEQ("\033[1;2F", MOD_SHIFT | KEY_END);
631 1 EXPECT_PARSE_SEQ("\033[1;2H", MOD_SHIFT | KEY_HOME);
632 1 EXPECT_PARSE_SEQ("\033[1;8H", MOD_SHIFT | MOD_META | MOD_CTRL | KEY_HOME);
633 1 EXPECT_PARSE_SEQN("\033[1;8H~", 6, MOD_SHIFT | MOD_META | MOD_CTRL | KEY_HOME);
634 1 EXPECT_PARSE_SEQN("\033[1;8H~_", 6, MOD_SHIFT | MOD_META | MOD_CTRL | KEY_HOME);
635 1 EXPECT_PARSE_SEQN("\033", TPARSE_PARTIAL_MATCH, 0);
636 1 EXPECT_PARSE_SEQN("\033[", TPARSE_PARTIAL_MATCH, 0);
637 1 EXPECT_PARSE_SEQN("\033]", TPARSE_PARTIAL_MATCH, 0);
638 1 EXPECT_PARSE_SEQN("\033[1", TPARSE_PARTIAL_MATCH, 0);
639 1 EXPECT_PARSE_SEQN("\033[9", TPARSE_PARTIAL_MATCH, 0);
640 1 EXPECT_PARSE_SEQN("\033[1;", TPARSE_PARTIAL_MATCH, 0);
641 1 EXPECT_PARSE_SEQN("\033[1[", 4, KEY_IGNORE);
642 1 EXPECT_PARSE_SEQN("\033[1;2", TPARSE_PARTIAL_MATCH, 0);
643 1 EXPECT_PARSE_SEQN("\033[1;8", TPARSE_PARTIAL_MATCH, 0);
644 1 EXPECT_PARSE_SEQN("\033[1;9", TPARSE_PARTIAL_MATCH, 0);
645 1 EXPECT_PARSE_SEQ("\033[1;_", KEY_IGNORE);
646 1 EXPECT_PARSE_SEQ("\033[1;8Z", KEY_IGNORE);
647 1 EXPECT_PARSE_SEQN("\033O", TPARSE_PARTIAL_MATCH, 0);
648 1 EXPECT_PARSE_SEQN("\033[\033", 2, KEY_IGNORE);
649 1 EXPECT_PARSE_SEQN("\033[\030", 3, KEY_IGNORE);
650 1 EXPECT_PARSE_SEQN("\033[1\030;2A", 4, KEY_IGNORE); // Cancelled by CAN
651 1 EXPECT_PARSE_SEQN("\033[1\031;2A", 7, MOD_SHIFT | KEY_UP); // EM ignored
652 1 EXPECT_PARSE_SEQN("\033[1\032;2A", 4, KEY_IGNORE); // Cancelled by SUB
653 1 EXPECT_PARSE_SEQN("\033[1\033;2A", 3, KEY_IGNORE); // Cancelled by (unconsumed) ESC
654 1 EXPECT_PARSE_SEQN("\033[1\177;2A", 7, MOD_SHIFT | KEY_UP); // DEL ignored
655 1 EXPECT_PARSE_SEQ("\033[A", KEY_UP);
656 1 EXPECT_PARSE_SEQ("\033[B", KEY_DOWN);
657 1 EXPECT_PARSE_SEQ("\033[C", KEY_RIGHT);
658 1 EXPECT_PARSE_SEQ("\033[D", KEY_LEFT);
659 1 EXPECT_PARSE_SEQ("\033[E", KEY_BEGIN);
660 1 EXPECT_PARSE_SEQ("\033[F", KEY_END);
661 1 EXPECT_PARSE_SEQ("\033[H", KEY_HOME);
662 1 EXPECT_PARSE_SEQ("\033[L", KEY_INSERT);
663 1 EXPECT_PARSE_SEQ("\033[P", KEY_F1);
664 1 EXPECT_PARSE_SEQ("\033[Q", KEY_F2);
665 1 EXPECT_PARSE_SEQ("\033[R", KEY_F3);
666 1 EXPECT_PARSE_SEQ("\033[S", KEY_F4);
667 1 EXPECT_PARSE_SEQ("\033O ", KEY_SPACE);
668 1 EXPECT_PARSE_SEQ("\033OA", KEY_UP);
669 1 EXPECT_PARSE_SEQ("\033OB", KEY_DOWN);
670 1 EXPECT_PARSE_SEQ("\033OC", KEY_RIGHT);
671 1 EXPECT_PARSE_SEQ("\033OD", KEY_LEFT);
672 1 EXPECT_PARSE_SEQ("\033OE", KEY_BEGIN);
673 1 EXPECT_PARSE_SEQ("\033OF", KEY_END);
674 1 EXPECT_PARSE_SEQ("\033OH", KEY_HOME);
675 1 EXPECT_PARSE_SEQ("\033OI", KEY_TAB);
676 1 EXPECT_PARSE_SEQ("\033OJ", KEY_IGNORE);
677 1 EXPECT_PARSE_SEQ("\033OM", KEY_ENTER);
678 1 EXPECT_PARSE_SEQ("\033OP", KEY_F1);
679 1 EXPECT_PARSE_SEQ("\033OQ", KEY_F2);
680 1 EXPECT_PARSE_SEQ("\033OR", KEY_F3);
681 1 EXPECT_PARSE_SEQ("\033OS", KEY_F4);
682 1 EXPECT_PARSE_SEQ("\033OT", KEY_IGNORE);
683 1 EXPECT_PARSE_SEQ("\033OX", '=');
684 1 EXPECT_PARSE_SEQ("\033Oi", KEY_IGNORE);
685 1 EXPECT_PARSE_SEQ("\033Oj", '*');
686 1 EXPECT_PARSE_SEQ("\033Ok", '+');
687 1 EXPECT_PARSE_SEQ("\033Ol", ',');
688 1 EXPECT_PARSE_SEQ("\033Om", '-');
689 1 EXPECT_PARSE_SEQ("\033On", '.');
690 1 EXPECT_PARSE_SEQ("\033Oo", '/');
691 1 EXPECT_PARSE_SEQ("\033Op", '0');
692 1 EXPECT_PARSE_SEQ("\033Oq", '1');
693 1 EXPECT_PARSE_SEQ("\033Or", '2');
694 1 EXPECT_PARSE_SEQ("\033Os", '3');
695 1 EXPECT_PARSE_SEQ("\033Ot", '4');
696 1 EXPECT_PARSE_SEQ("\033Ou", '5');
697 1 EXPECT_PARSE_SEQ("\033Ov", '6');
698 1 EXPECT_PARSE_SEQ("\033Ow", '7');
699 1 EXPECT_PARSE_SEQ("\033Ox", '8');
700 1 EXPECT_PARSE_SEQ("\033Oy", '9');
701 1 EXPECT_PARSE_SEQ("\033Oz", KEY_IGNORE);
702 1 EXPECT_PARSE_SEQ("\033O~", KEY_IGNORE);
703 1 EXPECT_PARSE_SEQ("\033[1~", KEY_HOME);
704 1 EXPECT_PARSE_SEQ("\033[2~", KEY_INSERT);
705 1 EXPECT_PARSE_SEQ("\033[3~", KEY_DELETE);
706 1 EXPECT_PARSE_SEQ("\033[4~", KEY_END);
707 1 EXPECT_PARSE_SEQ("\033[5~", KEY_PAGE_UP);
708 1 EXPECT_PARSE_SEQ("\033[6~", KEY_PAGE_DOWN);
709 1 EXPECT_PARSE_SEQ("\033[7~", KEY_HOME);
710 1 EXPECT_PARSE_SEQ("\033[8~", KEY_END);
711 1 EXPECT_PARSE_SEQ("\033[10~", KEY_IGNORE);
712 1 EXPECT_PARSE_SEQ("\033[11~", KEY_F1);
713 1 EXPECT_PARSE_SEQ("\033[12~", KEY_F2);
714 1 EXPECT_PARSE_SEQ("\033[13~", KEY_F3);
715 1 EXPECT_PARSE_SEQ("\033[14~", KEY_F4);
716 1 EXPECT_PARSE_SEQ("\033[15~", KEY_F5);
717 1 EXPECT_PARSE_SEQ("\033[16~", KEY_IGNORE);
718 1 EXPECT_PARSE_SEQ("\033[17~", KEY_F6);
719 1 EXPECT_PARSE_SEQ("\033[18~", KEY_F7);
720 1 EXPECT_PARSE_SEQ("\033[19~", KEY_F8);
721 1 EXPECT_PARSE_SEQ("\033[20~", KEY_F9);
722 1 EXPECT_PARSE_SEQ("\033[21~", KEY_F10);
723 1 EXPECT_PARSE_SEQ("\033[22~", KEY_IGNORE);
724 1 EXPECT_PARSE_SEQ("\033[23~", KEY_F11);
725 1 EXPECT_PARSE_SEQ("\033[24~", KEY_F12);
726 1 EXPECT_PARSE_SEQ("\033[25~", KEY_F13);
727 1 EXPECT_PARSE_SEQ("\033[26~", KEY_F14);
728 1 EXPECT_PARSE_SEQ("\033[27~", KEY_IGNORE);
729 1 EXPECT_PARSE_SEQ("\033[28~", KEY_F15);
730 1 EXPECT_PARSE_SEQ("\033[29~", KEY_F16);
731 1 EXPECT_PARSE_SEQ("\033[30~", KEY_IGNORE);
732 1 EXPECT_PARSE_SEQ("\033[31~", KEY_F17);
733 1 EXPECT_PARSE_SEQ("\033[34~", KEY_F20);
734 1 EXPECT_PARSE_SEQ("\033[35~", KEY_IGNORE);
735 1 EXPECT_PARSE_SEQ("\033[6;3~", MOD_META | KEY_PAGE_DOWN);
736 1 EXPECT_PARSE_SEQ("\033[6;5~", MOD_CTRL | KEY_PAGE_DOWN);
737 1 EXPECT_PARSE_SEQ("\033[6;8~", MOD_SHIFT | MOD_META | MOD_CTRL | KEY_PAGE_DOWN);
738
739 // xterm + `modifyOtherKeys` option
740 1 EXPECT_PARSE_SEQ("\033[27;5;9~", MOD_CTRL | KEY_TAB);
741 1 EXPECT_PARSE_SEQ("\033[27;5;27~", MOD_CTRL | KEY_ESCAPE);
742 1 EXPECT_PARSE_SEQ("\033[27;8;27~", MOD_CTRL | MOD_META | MOD_SHIFT | KEY_ESCAPE);
743 1 EXPECT_PARSE_SEQ("\033[27;1;27~", KEY_ESCAPE); // Tested in principle only; xterm never sends this
744 1 EXPECT_PARSE_SEQ("\033[27;5;13~", MOD_CTRL | KEY_ENTER);
745 1 EXPECT_PARSE_SEQ("\033[27;6;13~", MOD_CTRL | MOD_SHIFT | KEY_ENTER);
746 1 EXPECT_PARSE_SEQ("\033[27;2;127~", MOD_SHIFT | KEY_BACKSPACE);
747 1 EXPECT_PARSE_SEQ("\033[27;6;127~", MOD_CTRL | MOD_SHIFT | KEY_BACKSPACE);
748 1 EXPECT_PARSE_SEQ("\033[27;8;127~", MOD_CTRL | MOD_META | MOD_SHIFT | KEY_BACKSPACE);
749 1 EXPECT_PARSE_SEQ("\033[27;2;8~", MOD_SHIFT | KEY_BACKSPACE);
750 1 EXPECT_PARSE_SEQ("\033[27;6;8~", MOD_CTRL | MOD_SHIFT | KEY_BACKSPACE);
751 1 EXPECT_PARSE_SEQ("\033[27;8;8~", MOD_CTRL | MOD_META | MOD_SHIFT | KEY_BACKSPACE);
752 1 EXPECT_PARSE_SEQ("\033[27;6;82~", MOD_CTRL | MOD_SHIFT | 'r');
753 1 EXPECT_PARSE_SEQ("\033[27;5;114~", MOD_CTRL | 'r');
754 1 EXPECT_PARSE_SEQ("\033[27;3;82~", MOD_META | MOD_SHIFT | 'r'); // Sent by tmux 3.5-76-gbbc3cc55 (xterm sends "\033[27;4;82~")
755 1 EXPECT_PARSE_SEQ("\033[27;3;114~", MOD_META | 'r');
756 // EXPECT_PARSE_SEQ("\033[27;4;62~", MOD_META | '>');
757 1 EXPECT_PARSE_SEQ("\033[27;5;46~", MOD_CTRL | '.');
758 1 EXPECT_PARSE_SEQ("\033[27;3;1114111~", MOD_META | UNICODE_MAX_VALID_CODEPOINT);
759 1 EXPECT_PARSE_SEQ("\033[27;3;1114112~", KEY_IGNORE);
760 1 EXPECT_PARSE_SEQ("\033[27;999999999999999999999;123~", KEY_IGNORE);
761 1 EXPECT_PARSE_SEQ("\033[27;123;99999999999999999~", KEY_IGNORE);
762
763 // www.leonerd.org.uk/hacks/fixterms/
764 1 EXPECT_PARSE_SEQ("\033[13;3u", MOD_META | KEY_ENTER);
765 1 EXPECT_PARSE_SEQ("\033[9;5u", MOD_CTRL | KEY_TAB);
766 1 EXPECT_PARSE_SEQ("\033[65;3u", MOD_META | 'A');
767 1 EXPECT_PARSE_SEQ("\033[108;5u", MOD_CTRL | 'l');
768 1 EXPECT_PARSE_SEQ("\033[127765;3u", MOD_META | 127765ul);
769 1 EXPECT_PARSE_SEQ("\033[1114111;3u", MOD_META | UNICODE_MAX_VALID_CODEPOINT);
770 1 EXPECT_PARSE_SEQ("\033[1114112;3u", KEY_IGNORE);
771 1 EXPECT_PARSE_SEQ("\033[11141110;3u", KEY_IGNORE);
772 1 EXPECT_PARSE_SEQ("\033[11141111;3u", KEY_IGNORE);
773 1 EXPECT_PARSE_SEQ("\033[2147483647;3u", KEY_IGNORE); // INT32_MAX
774 1 EXPECT_PARSE_SEQ("\033[2147483648;3u", KEY_IGNORE); // INT32_MAX + 1
775 1 EXPECT_PARSE_SEQ("\033[4294967295;3u", KEY_IGNORE); // UINT32_MAX
776 1 EXPECT_PARSE_SEQ("\033[4294967296;3u", KEY_IGNORE); // UINT32_MAX + 1
777 1 EXPECT_PARSE_SEQ("\033[-1;3u", KEY_IGNORE);
778 1 EXPECT_PARSE_SEQ("\033[-2;3u", KEY_IGNORE);
779 1 EXPECT_PARSE_SEQ("\033[ 2;3u", KEY_IGNORE);
780 1 EXPECT_PARSE_SEQ("\033[<?>2;3u", KEY_IGNORE);
781 1 EXPECT_PARSE_SEQ("\033[ !//.$2;3u", KEY_IGNORE);
782
783 // https://sw.kovidgoyal.net/kitty/keyboard-protocol
784 1 EXPECT_PARSE_SEQ("\033[27u", KEY_ESCAPE);
785 1 EXPECT_PARSE_SEQ("\033[27;5u", MOD_CTRL | KEY_ESCAPE);
786 1 EXPECT_PARSE_SEQ("\033[27;8u", MOD_CTRL | MOD_META | MOD_SHIFT | KEY_ESCAPE);
787 1 EXPECT_PARSE_SEQ("\033[57359u", KEY_SCROLL_LOCK);
788 1 EXPECT_PARSE_SEQ("\033[57360u", KEY_IGNORE);
789 1 EXPECT_PARSE_SEQ("\033[57361u", KEY_PRINT_SCREEN);
790 1 EXPECT_PARSE_SEQ("\033[57362u", KEY_PAUSE);
791 1 EXPECT_PARSE_SEQ("\033[57363u", KEY_MENU);
792 1 EXPECT_PARSE_SEQ("\033[57376u", KEY_F13);
793 1 EXPECT_PARSE_SEQ("\033[57377u", KEY_F14);
794 1 EXPECT_PARSE_SEQ("\033[57378u", KEY_F15);
795 1 EXPECT_PARSE_SEQ("\033[57379u", KEY_F16);
796 1 EXPECT_PARSE_SEQ("\033[57380u", KEY_F17);
797 1 EXPECT_PARSE_SEQ("\033[57381u", KEY_F18);
798 1 EXPECT_PARSE_SEQ("\033[57382u", KEY_F19);
799 1 EXPECT_PARSE_SEQ("\033[57383u", KEY_F20);
800 1 EXPECT_PARSE_SEQ("\033[57384u", KEY_F21);
801 1 EXPECT_PARSE_SEQ("\033[57385u", KEY_F22);
802 1 EXPECT_PARSE_SEQ("\033[57386u", KEY_F23);
803 1 EXPECT_PARSE_SEQ("\033[57387u", KEY_F24);
804 1 EXPECT_PARSE_SEQ("\033[57388u", KEY_F25);
805 1 EXPECT_PARSE_SEQ("\033[57389u", KEY_F26);
806 1 EXPECT_PARSE_SEQ("\033[57390u", KEY_F27);
807 1 EXPECT_PARSE_SEQ("\033[57391u", KEY_F28);
808 1 EXPECT_PARSE_SEQ("\033[57392u", KEY_F29);
809 1 EXPECT_PARSE_SEQ("\033[57393u", KEY_F30);
810 1 EXPECT_PARSE_SEQ("\033[57394u", KEY_F31);
811 1 EXPECT_PARSE_SEQ("\033[57395u", KEY_F32);
812 1 EXPECT_PARSE_SEQ("\033[57396u", KEY_F33);
813 1 EXPECT_PARSE_SEQ("\033[57397u", KEY_F34);
814 1 EXPECT_PARSE_SEQ("\033[57398u", KEY_F35);
815 1 EXPECT_PARSE_SEQ("\033[57399u", '0');
816 1 EXPECT_PARSE_SEQ("\033[57400u", '1');
817 1 EXPECT_PARSE_SEQ("\033[57401u", '2');
818 1 EXPECT_PARSE_SEQ("\033[57402u", '3');
819 1 EXPECT_PARSE_SEQ("\033[57403u", '4');
820 1 EXPECT_PARSE_SEQ("\033[57404u", '5');
821 1 EXPECT_PARSE_SEQ("\033[57405u", '6');
822 1 EXPECT_PARSE_SEQ("\033[57406u", '7');
823 1 EXPECT_PARSE_SEQ("\033[57407u", '8');
824 1 EXPECT_PARSE_SEQ("\033[57408u", '9');
825 1 EXPECT_PARSE_SEQ("\033[57409u", '.');
826 1 EXPECT_PARSE_SEQ("\033[57410u", '/');
827 1 EXPECT_PARSE_SEQ("\033[57411u", '*');
828 1 EXPECT_PARSE_SEQ("\033[57412u", '-');
829 1 EXPECT_PARSE_SEQ("\033[57413u", '+');
830 1 EXPECT_PARSE_SEQ("\033[57414u", KEY_ENTER);
831 1 EXPECT_PARSE_SEQ("\033[57415u", '=');
832 1 EXPECT_PARSE_SEQ("\033[57416u", KEY_IGNORE); // KP_SEPARATOR
833 1 EXPECT_PARSE_SEQ("\033[57417u", KEY_LEFT);
834 1 EXPECT_PARSE_SEQ("\033[57418u", KEY_RIGHT);
835 1 EXPECT_PARSE_SEQ("\033[57419u", KEY_UP);
836 1 EXPECT_PARSE_SEQ("\033[57420u", KEY_DOWN);
837 1 EXPECT_PARSE_SEQ("\033[57421u", KEY_PAGE_UP);
838 1 EXPECT_PARSE_SEQ("\033[57422u", KEY_PAGE_DOWN);
839 1 EXPECT_PARSE_SEQ("\033[57423u", KEY_HOME);
840 1 EXPECT_PARSE_SEQ("\033[57424u", KEY_END);
841 1 EXPECT_PARSE_SEQ("\033[57425u", KEY_INSERT);
842 1 EXPECT_PARSE_SEQ("\033[57426u", KEY_DELETE);
843 1 EXPECT_PARSE_SEQ("\033[57427u", KEY_BEGIN);
844 1 EXPECT_PARSE_SEQ("\033[57361;2u", MOD_SHIFT | KEY_PRINT_SCREEN);
845 1 EXPECT_PARSE_SEQ("\033[3615:3620:97;6u", MOD_CTRL | MOD_SHIFT | 'a');
846 1 EXPECT_PARSE_SEQ("\033[97;5u", MOD_CTRL | 'a');
847 1 EXPECT_PARSE_SEQ("\033[97;5:1u", MOD_CTRL | 'a');
848 1 EXPECT_PARSE_SEQ("\033[97;5:2u", MOD_CTRL | 'a');
849 1 EXPECT_PARSE_SEQ("\033[97;5:3u", KEY_IGNORE);
850 // TODO: EXPECT_PARSE_SEQ("\033[97;5:u", MOD_CTRL | 'a');
851 1 EXPECT_PARSE_SEQ("\033[104;5u", MOD_CTRL | 'h');
852 1 EXPECT_PARSE_SEQ("\033[104;9u", MOD_SUPER | 'h');
853 1 EXPECT_PARSE_SEQ("\033[104;17u", MOD_HYPER | 'h');
854 1 EXPECT_PARSE_SEQ("\033[104;10u", MOD_SUPER | MOD_SHIFT | 'h');
855 1 EXPECT_PARSE_SEQ("\033[116;69u", MOD_CTRL | 't'); // Ignored Capslock in modifiers
856 1 EXPECT_PARSE_SEQ("\033[116;133u", MOD_CTRL | 't'); // Ignored Numlock in modifiers
857 1 EXPECT_PARSE_SEQ("\033[116;256u", MOD_MASK | 't'); // Ignored Capslock and Numlock
858 1 EXPECT_PARSE_SEQ("\033[116;257u", KEY_IGNORE); // Unknown bit in modifiers
859
860 // Excess params
861 1 EXPECT_PARSE_SEQ("\033[1;2;3;4;5;6;7;8;9m", KEY_IGNORE);
862
863 // DA1 replies
864 1 EXPECT_PARSE_SEQ("\033[?64;15;22c", TFLAG(TFLAG_QUERY_L2 | TFLAG_QUERY_L3 | TFLAG_8_COLOR));
865 1 EXPECT_PARSE_SEQ("\033[?1;2c", TFLAG(TFLAG_QUERY_L2));
866 1 EXPECT_PARSE_SEQ("\033[?90c", KEY_IGNORE);
867
868 // DA2 replies
869 1 EXPECT_PARSE_SEQ("\033[>0;136;0c", KEY_IGNORE);
870 1 EXPECT_PARSE_SEQ("\033[>84;0;0c", TFLAG(TFLAG_QUERY_L3)); // tmux
871 1 EXPECT_PARSE_SEQ("\033[>1;11801;0c", TFLAG(TFLAG_QUERY_L3)); // foot
872
873 // DA3 replies
874 1 EXPECT_PARSE_SEQN("\033P!|464f4f54\033\\", 12, TFLAG(TFLAG_QUERY_L3)); // foot
875 1 EXPECT_PARSE_SEQN("\033P!|464f4f54\033", 12, TFLAG(TFLAG_QUERY_L3));
876 1 EXPECT_PARSE_SEQN("\033P!|464f4f54", TPARSE_PARTIAL_MATCH, 0);
877
878 // XTVERSION replies
879 1 const TermFeatureFlags tmuxflags = TFLAG (
880 TFLAG_NO_QUERY_L3 | TFLAG_ECMA48_REPEAT | TFLAG_MODIFY_OTHER_KEYS
881 | TFLAG_BS_CTRL_BACKSPACE
882 );
883 1 EXPECT_PARSE_SEQN("\033P>|tmux 3.2\033\\", 12, tmuxflags);
884 1 EXPECT_PARSE_SEQN("\033P>|tmux 3.2a\033\\", 13, tmuxflags);
885 1 EXPECT_PARSE_SEQN("\033P>|tmux 3.2-rc2\033\\", 16, tmuxflags);
886 1 EXPECT_PARSE_SEQN("\033P>|tmux next-3.3\033\\", 17, tmuxflags);
887 1 EXPECT_PARSE_SEQN("\033P>|xyz\033\\", 7, TFLAG(TFLAG_QUERY_L3));
888
889 // XTMODKEYS (modifyOtherKeys mode query) replies
890 1 EXPECT_PARSE_SEQ("\033[>4;0m", TFLAG(TFLAG_MODIFY_OTHER_KEYS));
891 1 EXPECT_PARSE_SEQ("\033[>4;1m", TFLAG(TFLAG_MODIFY_OTHER_KEYS));
892 1 EXPECT_PARSE_SEQ("\033[>4;2m", TFLAG(TFLAG_MODIFY_OTHER_KEYS));
893 1 EXPECT_PARSE_SEQ("\033[>4;3m", KEY_IGNORE);
894 1 EXPECT_PARSE_SEQ("\033[>4;m", TFLAG(TFLAG_MODIFY_OTHER_KEYS));
895 1 EXPECT_PARSE_SEQ("\033[>4m", TFLAG(TFLAG_MODIFY_OTHER_KEYS));
896 1 EXPECT_PARSE_SEQ("\033[>4;2;0m", KEY_IGNORE);
897
898 // XTWINOPS replies
899 1 EXPECT_PARSE_SEQN("\033]ltitle\033\\", 8, KEY_IGNORE);
900 1 EXPECT_PARSE_SEQN("\033]Licon\033\\", 7, KEY_IGNORE);
901 1 EXPECT_PARSE_SEQ("\033]ltitle\a", KEY_IGNORE);
902 1 EXPECT_PARSE_SEQ("\033]Licon\a", KEY_IGNORE);
903
904 // DECRPM replies (to DECRQM queries)
905 1 EXPECT_PARSE_SEQ("\033[?2026;0$y", KEY_IGNORE);
906 1 EXPECT_PARSE_SEQ("\033[?2026;1$y", TFLAG(TFLAG_SYNC));
907 1 EXPECT_PARSE_SEQ("\033[?2026;2$y", TFLAG(TFLAG_SYNC));
908 1 EXPECT_PARSE_SEQ("\033[?2026;3$y", KEY_IGNORE);
909 1 EXPECT_PARSE_SEQ("\033[?2026;4$y", KEY_IGNORE);
910 1 EXPECT_PARSE_SEQ("\033[?2026;5$y", KEY_IGNORE);
911 1 EXPECT_PARSE_SEQ("\033[?0;1$y", KEY_IGNORE);
912 1 EXPECT_PARSE_SEQ("\033[?1036;2$y", TFLAG(TFLAG_META_ESC));
913 1 EXPECT_PARSE_SEQ("\033[?1039;2$y", TFLAG(TFLAG_ALT_ESC));
914 1 EXPECT_PARSE_SEQ("\033[?7;2$y", KEY_IGNORE);
915 1 EXPECT_PARSE_SEQ("\033[?25;2$y", KEY_IGNORE);
916 1 EXPECT_PARSE_SEQ("\033[?45;2$y", KEY_IGNORE);
917 1 EXPECT_PARSE_SEQ("\033[?67;2$y", KEY_IGNORE);
918 1 EXPECT_PARSE_SEQ("\033[?1049;2$y", KEY_IGNORE);
919 1 EXPECT_PARSE_SEQ("\033[?2004;2$y", KEY_IGNORE);
920
921 // Invalid, DECRPM-like sequences
922 1 EXPECT_PARSE_SEQ("\033[?9$y", KEY_IGNORE); // Too few params
923 1 EXPECT_PARSE_SEQ("\033[?1;2;3$y", KEY_IGNORE); // Too many params
924 1 EXPECT_PARSE_SEQ("\033[?1;2y", KEY_IGNORE); // No '$' intermediate byte
925 1 EXPECT_PARSE_SEQ("\033[1;2$y", KEY_IGNORE); // No '?' param prefix
926
927 // XTGETTCAP replies
928 1 EXPECT_PARSE_SEQN("\033P1+r626365\033\\", 11, TFLAG(TFLAG_BACK_COLOR_ERASE));
929 1 EXPECT_PARSE_SEQN("\033P1+r74736C=1B5D323B\033\\", 20, TFLAG(TFLAG_SET_WINDOW_TITLE));
930 1 EXPECT_PARSE_SEQN("\033P0+r\033\\", 5, KEY_IGNORE);
931 1 EXPECT_PARSE_SEQN("\033P0+rbbccdd\033\\", 11, KEY_IGNORE);
932
933 // DECRQSS replies
934 1 EXPECT_PARSE_SEQN("\033P1$r0;38:2::60:70:80;48:5:255m\033\\", 31, TFLAG(TFLAG_TRUE_COLOR | TFLAG_256_COLOR)); // SGR (xterm, foot)
935 1 EXPECT_PARSE_SEQN("\033P1$r0;38:2:60:70:80;48:5:255m\033\\", 30, TFLAG(TFLAG_TRUE_COLOR | TFLAG_256_COLOR)); // SGR (kitty)
936 1 EXPECT_PARSE_SEQN("\033P1$r0;zm\033\\", 9, KEY_IGNORE); // Invalid SGR-like
937 1 EXPECT_PARSE_SEQN("\033P1$r2 q\033\\", 8, KEY_IGNORE); // DECSCUSR 2 (cursor style)
938
939 // Kitty keyboard protocol query replies
940 1 EXPECT_PARSE_SEQ("\033[?5u", TFLAG(TFLAG_KITTY_KEYBOARD));
941 1 EXPECT_PARSE_SEQ("\033[?1u", TFLAG(TFLAG_KITTY_KEYBOARD));
942 1 EXPECT_PARSE_SEQ("\033[?0u", TFLAG(TFLAG_KITTY_KEYBOARD));
943
944 // Invalid, kitty-reply-like sequences
945 1 EXPECT_PARSE_SEQ("\033[?u", KEY_IGNORE); // Too few params (could be valid, in theory)
946 1 EXPECT_PARSE_SEQ("\033[?1;2u", KEY_IGNORE); // Too many params
947 1 EXPECT_PARSE_SEQ("\033[?1:2u", KEY_IGNORE); // Sub-params
948 1 EXPECT_PARSE_SEQ("\033[?1!u", KEY_IGNORE); // Intermediate '!' byte
949 1 }
950
951 1 static void test_term_parse_sequence2(TestContext *ctx)
952 {
953 1 static const struct {
954 char key_str[3];
955 char final_byte;
956 KeyCode key;
957 } templates[] = {
958 {"1", 'A', KEY_UP},
959 {"1", 'B', KEY_DOWN},
960 {"1", 'C', KEY_RIGHT},
961 {"1", 'D', KEY_LEFT},
962 {"1", 'E', KEY_BEGIN},
963 {"1", 'F', KEY_END},
964 {"1", 'H', KEY_HOME},
965 {"1", 'P', KEY_F1},
966 {"1", 'Q', KEY_F2},
967 {"1", 'R', KEY_F3}, // https://sw.kovidgoyal.net/kitty/keyboard-protocol/#:~:text=CSI%20R
968 {"1", 'S', KEY_F4},
969 {"2", '~', KEY_INSERT},
970 {"3", '~', KEY_DELETE},
971 {"5", '~', KEY_PAGE_UP},
972 {"6", '~', KEY_PAGE_DOWN},
973 {"11", '~', KEY_F1},
974 {"12", '~', KEY_F2},
975 {"13", '~', KEY_F3},
976 {"14", '~', KEY_F4},
977 {"15", '~', KEY_F5},
978 {"17", '~', KEY_F6},
979 {"18", '~', KEY_F7},
980 {"19", '~', KEY_F8},
981 {"20", '~', KEY_F9},
982 {"21", '~', KEY_F10},
983 {"23", '~', KEY_F11},
984 {"24", '~', KEY_F12},
985 {"25", '~', KEY_F13},
986 {"26", '~', KEY_F14},
987 {"28", '~', KEY_F15},
988 {"29", '~', KEY_F16},
989 {"31", '~', KEY_F17},
990 {"32", '~', KEY_F18},
991 {"33", '~', KEY_F19},
992 {"34", '~', KEY_F20},
993 };
994
995 1 static const struct {
996 char mod_str[4];
997 KeyCode mask;
998 } modifiers[] = {
999 {"0", KEY_IGNORE},
1000 {"1", 0},
1001 {"2", MOD_SHIFT},
1002 {"3", MOD_META},
1003 {"4", MOD_SHIFT | MOD_META},
1004 {"5", MOD_CTRL},
1005 {"6", MOD_SHIFT | MOD_CTRL},
1006 {"7", MOD_META | MOD_CTRL},
1007 {"8", MOD_SHIFT | MOD_META | MOD_CTRL},
1008 {"9", MOD_SUPER},
1009 {"10", MOD_SUPER | MOD_SHIFT},
1010 {"11", MOD_SUPER | MOD_META},
1011 {"12", MOD_SUPER | MOD_META | MOD_SHIFT},
1012 {"13", MOD_SUPER | MOD_CTRL},
1013 {"14", MOD_SUPER | MOD_CTRL | MOD_SHIFT},
1014 {"15", MOD_SUPER | MOD_META | MOD_CTRL},
1015 {"16", MOD_SUPER | MOD_META | MOD_CTRL | MOD_SHIFT},
1016 {"17", MOD_HYPER},
1017 {"18", MOD_HYPER | MOD_SHIFT},
1018 {"255", MOD_HYPER | MOD_SUPER | MOD_META | MOD_CTRL},
1019 {"256", MOD_MASK},
1020 {"257", KEY_IGNORE},
1021 {"400", KEY_IGNORE},
1022 };
1023
1024 1 static_assert(KEY_IGNORE != 0);
1025 1 static_assert((KEY_IGNORE & MOD_MASK) == 0);
1026
1027
2/2
✓ Branch 21 → 22 taken 35 times.
✓ Branch 21 → 23 taken 1 time.
36 FOR_EACH_I(i, templates) {
1028
2/2
✓ Branch 19 → 3 taken 805 times.
✓ Branch 19 → 20 taken 35 times.
840 FOR_EACH_I(j, modifiers) {
1029 805 char seq[16];
1030 1610 size_t seq_length = xsnprintf (
1031 seq,
1032 sizeof seq,
1033 "\033[%s;%s%c",
1034 805 templates[i].key_str,
1035 805 modifiers[j].mod_str,
1036 805 templates[i].final_byte
1037 );
1038 805 KeyCode key = 24;
1039 805 ssize_t parsed_length = term_parse_sequence(seq, seq_length, &key);
1040 805 KeyCode mods = modifiers[j].mask;
1041
2/2
✓ Branch 5 → 6 taken 700 times.
✓ Branch 5 → 7 taken 105 times.
805 KeyCode expected_key = mods | (mods == KEY_IGNORE ? 0 : templates[i].key);
1042 805 IEXPECT_EQ(parsed_length, seq_length);
1043 805 IEXPECT_KEYCODE_EQ(key, expected_key, seq, seq_length);
1044 // Truncated
1045 805 key = 25;
1046
2/2
✓ Branch 14 → 10 taken 5080 times.
✓ Branch 14 → 15 taken 805 times.
5885 for (size_t n = seq_length - 1; n != 0; n--) {
1047 5080 parsed_length = term_parse_sequence(seq, n, &key);
1048 5080 EXPECT_EQ(parsed_length, TPARSE_PARTIAL_MATCH);
1049 5080 EXPECT_EQ(key, 25);
1050 }
1051 // Overlength
1052 805 key = 26;
1053 805 seq[seq_length++] = '~';
1054 805 parsed_length = term_parse_sequence(seq, seq_length, &key);
1055 805 IEXPECT_EQ(parsed_length, seq_length - 1);
1056 805 IEXPECT_KEYCODE_EQ(key, expected_key, seq, seq_length);
1057 }
1058 }
1059 1 }
1060
1061 1 static void test_rxvt_parse_key(TestContext *ctx)
1062 {
1063 1 static const struct {
1064 char escape_sequence[8];
1065 KeyCode key;
1066 } templates[] = {
1067 {"\033\033[2_", KEY_INSERT},
1068 {"\033\033[3_", KEY_DELETE},
1069 {"\033\033[5_", KEY_PAGE_UP},
1070 {"\033\033[6_", KEY_PAGE_DOWN},
1071 {"\033\033[7_", KEY_HOME},
1072 {"\033\033[8_", KEY_END},
1073 };
1074
1075 1 static const struct {
1076 char ch;
1077 KeyCode mask;
1078 } modifiers[] = {
1079 {'~', 0},
1080 {'^', MOD_CTRL},
1081 {'$', MOD_SHIFT},
1082 {'@', MOD_SHIFT | MOD_CTRL},
1083 };
1084
1085
2/2
✓ Branch 26 → 24 taken 6 times.
✓ Branch 26 → 33 taken 1 time.
7 FOR_EACH_I(i, templates) {
1086
2/2
✓ Branch 24 → 3 taken 24 times.
✓ Branch 24 → 25 taken 6 times.
30 FOR_EACH_I(j, modifiers) {
1087 24 char seq[8];
1088 24 memcpy(seq, templates[i].escape_sequence, 8);
1089 24 ASSERT_EQ(seq[7], '\0');
1090 24 char *underscore = strchr(seq, '_');
1091 24 ASSERT_NONNULL(underscore);
1092 24 *underscore = modifiers[j].ch;
1093 24 size_t seq_length = strlen(seq);
1094 24 KeyCode key;
1095
1096 // Double ESC
1097 24 ssize_t parsed_length = rxvt_parse_key(seq, seq_length, &key);
1098 24 EXPECT_EQ(parsed_length, seq_length);
1099 24 EXPECT_EQ(key, modifiers[j].mask | templates[i].key | MOD_META);
1100
1101 // Single ESC
1102 24 parsed_length = rxvt_parse_key(seq + 1, seq_length - 1, &key);
1103 24 EXPECT_EQ(parsed_length, seq_length - 1);
1104 24 EXPECT_EQ(key, modifiers[j].mask | templates[i].key);
1105
1106 // Truncated (double ESC)
1107
2/2
✓ Branch 16 → 12 taken 96 times.
✓ Branch 16 → 17 taken 24 times.
144 for (size_t n = seq_length - 1; n != 0; n--) {
1108 96 key = 25;
1109 96 parsed_length = rxvt_parse_key(seq, n, &key);
1110 96 EXPECT_EQ(parsed_length, TPARSE_PARTIAL_MATCH);
1111 96 EXPECT_EQ(key, 25);
1112 }
1113
1114 // Truncated (single ESC)
1115
2/2
✓ Branch 22 → 18 taken 72 times.
✓ Branch 22 → 23 taken 24 times.
96 for (size_t n = seq_length - 2; n != 0; n--) {
1116 72 key = 25;
1117 72 parsed_length = rxvt_parse_key(seq + 1, n, &key);
1118 72 EXPECT_EQ(parsed_length, TPARSE_PARTIAL_MATCH);
1119 72 EXPECT_EQ(key, 25);
1120 }
1121 }
1122 }
1123
1124 static const struct {
1125 char NONSTRING seq[6];
1126 uint8_t seq_length;
1127 int8_t expected_length;
1128 KeyCode expected_key;
1129 } tests[] = {
1130 {STRN("\033Oa"), 3, MOD_CTRL | KEY_UP},
1131 {STRN("\033Ob"), 3, MOD_CTRL | KEY_DOWN},
1132 {STRN("\033Oc"), 3, MOD_CTRL | KEY_RIGHT},
1133 {STRN("\033Od"), 3, MOD_CTRL | KEY_LEFT},
1134 {STRN("\033O"), TPARSE_PARTIAL_MATCH, 0},
1135 {STRN("\033[a"), 3, MOD_SHIFT | KEY_UP},
1136 {STRN("\033[b"), 3, MOD_SHIFT | KEY_DOWN},
1137 {STRN("\033[c"), 3, MOD_SHIFT | KEY_RIGHT},
1138 {STRN("\033[d"), 3, MOD_SHIFT | KEY_LEFT},
1139 {STRN("\033["), TPARSE_PARTIAL_MATCH, 0},
1140 {STRN("\033[1;5A"), 6, MOD_CTRL | KEY_UP},
1141 {STRN("\033[1;5"), TPARSE_PARTIAL_MATCH, 0},
1142 {STRN("\033\033[@"), 4, KEY_IGNORE},
1143 {STRN("\033\033["), TPARSE_PARTIAL_MATCH, 0},
1144 {STRN("\033\033"), TPARSE_PARTIAL_MATCH, 0},
1145 {STRN("\033"), TPARSE_PARTIAL_MATCH, 0},
1146 };
1147
1148
2/2
✓ Branch 33 → 27 taken 16 times.
✓ Branch 33 → 34 taken 1 time.
17 FOR_EACH_I(i, tests) {
1149 16 const char *seq = tests[i].seq;
1150 16 const size_t seq_length = tests[i].seq_length;
1151 16 KeyCode key = 0x18;
1152 16 ssize_t parsed_length = rxvt_parse_key(seq, seq_length, &key);
1153
2/2
✓ Branch 28 → 29 taken 10 times.
✓ Branch 28 → 30 taken 6 times.
16 KeyCode expected_key = (parsed_length > 0) ? tests[i].expected_key : 0x18;
1154 16 IEXPECT_EQ(parsed_length, tests[i].expected_length);
1155 16 IEXPECT_KEYCODE_EQ(key, expected_key, seq, seq_length);
1156 }
1157 1 }
1158
1159 1 static void test_linux_parse_key(TestContext *ctx)
1160 {
1161 1 static const struct {
1162 char NONSTRING seq[6];
1163 uint8_t seq_length;
1164 int8_t expected_length;
1165 KeyCode expected_key;
1166 } tests[] = {
1167 {STRN("\033[1;5A"), 6, MOD_CTRL | KEY_UP},
1168 {STRN("\033[[A"), 4, KEY_F1},
1169 {STRN("\033[[B"), 4, KEY_F2},
1170 {STRN("\033[[C"), 4, KEY_F3},
1171 {STRN("\033[[D"), 4, KEY_F4},
1172 {STRN("\033[[E"), 4, KEY_F5},
1173 {STRN("\033[[F"), 3, KEY_IGNORE},
1174 {STRN("\033[["), TPARSE_PARTIAL_MATCH, 0},
1175 {STRN("\033["), TPARSE_PARTIAL_MATCH, 0},
1176 {STRN("\033"), TPARSE_PARTIAL_MATCH, 0},
1177 {STRN(""), TPARSE_PARTIAL_MATCH, 0},
1178 };
1179
1180
2/2
✓ Branch 9 → 3 taken 11 times.
✓ Branch 9 → 10 taken 1 time.
12 FOR_EACH_I(i, tests) {
1181 11 const char *seq = tests[i].seq;
1182 11 const size_t seq_length = tests[i].seq_length;
1183 11 KeyCode key = 0x18;
1184 11 ssize_t parsed_length = linux_parse_key(seq, seq_length, &key);
1185
2/2
✓ Branch 4 → 5 taken 7 times.
✓ Branch 4 → 6 taken 4 times.
11 KeyCode expected_key = (parsed_length > 0) ? tests[i].expected_key : 0x18;
1186 11 IEXPECT_EQ(parsed_length, tests[i].expected_length);
1187 11 IEXPECT_KEYCODE_EQ(key, expected_key, seq, seq_length);
1188 }
1189 1 }
1190
1191 1 static void test_keycode_to_string(TestContext *ctx)
1192 {
1193 1 static const struct {
1194 const char *str;
1195 KeyCode key;
1196 } tests[] = {
1197 {"a", 'a'},
1198 {"Z", 'Z'},
1199 {"0", '0'},
1200 {"{", '{'},
1201 {"space", KEY_SPACE},
1202 {"enter", KEY_ENTER},
1203 {"tab", KEY_TAB},
1204 {"escape", KEY_ESCAPE},
1205 {"backspace", KEY_BACKSPACE},
1206 {"insert", KEY_INSERT},
1207 {"delete", KEY_DELETE},
1208 {"home", KEY_HOME},
1209 {"end", KEY_END},
1210 {"pgup", KEY_PAGE_UP},
1211 {"pgdown", KEY_PAGE_DOWN},
1212 {"begin", KEY_BEGIN},
1213 {"left", KEY_LEFT},
1214 {"right", KEY_RIGHT},
1215 {"up", KEY_UP},
1216 {"down", KEY_DOWN},
1217 {"scrlock", KEY_SCROLL_LOCK},
1218 {"print", KEY_PRINT_SCREEN},
1219 {"pause", KEY_PAUSE},
1220 {"menu", KEY_MENU},
1221 {"C-a", MOD_CTRL | 'a'},
1222 {"C-s", MOD_CTRL | 's'},
1223 {"M-a", MOD_META | 'a'},
1224 {"M-S-{", MOD_META | MOD_SHIFT | '{'},
1225 {"C-S-a", MOD_CTRL | MOD_SHIFT | 'a'},
1226 {"M-S-a", MOD_META | MOD_SHIFT | 'a'},
1227 {"F1", KEY_F1},
1228 {"F5", KEY_F5},
1229 {"F12", KEY_F12},
1230 {"F13", KEY_F13},
1231 {"F16", KEY_F16},
1232 {"F20", KEY_F20},
1233 {"F24", KEY_F24},
1234 {"F35", KEY_F35},
1235 {"M-enter", MOD_META | KEY_ENTER},
1236 {"M-space", MOD_META | KEY_SPACE},
1237 {"S-tab", MOD_SHIFT | KEY_TAB},
1238 {"C-M-S-F12", MOD_CTRL | MOD_META | MOD_SHIFT | KEY_F12},
1239 {"C-M-F16", MOD_CTRL | MOD_META | KEY_F16},
1240 {"C-S-F20", MOD_CTRL | MOD_SHIFT | KEY_F20},
1241 {"C-S-F24", MOD_CTRL | MOD_SHIFT | KEY_F24},
1242 {"C-S-F35", MOD_CTRL | MOD_SHIFT | KEY_F35},
1243 {"C-M-S-up", MOD_CTRL | MOD_META | MOD_SHIFT | KEY_UP},
1244 {"C-M-delete", MOD_CTRL | MOD_META | KEY_DELETE},
1245 {"C-home", MOD_CTRL | KEY_HOME},
1246 {"s-space", MOD_SUPER | ' '},
1247 {"H-end", MOD_HYPER | KEY_END},
1248 #if __STDC_VERSION__ >= 201112L
1249 {(const char*)u8"ก", 0x0E01},
1250 {(const char*)u8"C-ก", MOD_CTRL | 0x0E01},
1251 {(const char*)u8"M-Ф", MOD_META | 0x0424},
1252 #endif
1253 };
1254
1255 1 char buf[KEYCODE_STR_BUFSIZE];
1256 1 ASSERT_TRUE(sizeof(buf) >= sizeof("QUERY REPLY; 0x12345678"));
1257 1 EXPECT_EQ(KEYCODE_STR_BUFSIZE, next_pow2(sizeof("QUERY REPLY; 0x12345678")));
1258
1259
2/2
✓ Branch 12 → 5 taken 54 times.
✓ Branch 12 → 13 taken 1 time.
56 FOR_EACH_I(i, tests) {
1260 54 const char *str = tests[i].str;
1261 54 size_t len = strlen(str);
1262 54 ASSERT_TRUE(len < sizeof(buf));
1263 54 size_t buflen = keycode_to_str(tests[i].key, buf);
1264 54 IEXPECT_STREQ(buf, str);
1265 54 IEXPECT_EQ(buflen, len);
1266 54 IEXPECT_EQ(keycode_from_str(str), tests[i].key);
1267 }
1268
1269 1 EXPECT_EQ(keycode_to_str(KEYCODE_DETECTED_PASTE, buf), 19);
1270 1 EXPECT_STREQ(buf, "INVALID; 0x00110034");
1271 1 EXPECT_EQ(keycode_to_str(KEYCODE_BRACKETED_PASTE, buf), 19);
1272 1 EXPECT_STREQ(buf, "INVALID; 0x00110035");
1273 1 EXPECT_EQ(keycode_to_str(KEY_IGNORE, buf), 19);
1274 1 EXPECT_STREQ(buf, "INVALID; 0x00110036");
1275 1 EXPECT_EQ(keycode_to_str(UINT32_MAX, buf), 23);
1276 1 EXPECT_STREQ(buf, "QUERY REPLY; 0xFFFFFFFF");
1277
1278 // These combos aren't round-trippable by the code above and can't end
1279 // up in real bindings, since the letters are normalized to lower case
1280 // by keycode_from_str(). We still test them nevertheless; for the sake
1281 // of completeness and catching unexpected changes.
1282 1 EXPECT_EQ(keycode_to_str(MOD_CTRL | 'A', buf), 3);
1283 1 EXPECT_STREQ(buf, "C-A");
1284 1 EXPECT_EQ(keycode_to_str(MOD_CTRL | MOD_SHIFT | 'A', buf), 5);
1285 1 EXPECT_STREQ(buf, "C-S-A");
1286 1 EXPECT_EQ(keycode_to_str(MOD_META | MOD_SHIFT | 'A', buf), 5);
1287 1 EXPECT_STREQ(buf, "M-S-A");
1288 1 }
1289
1290 1 static void test_parse_key_string(TestContext *ctx)
1291 {
1292 1 EXPECT_EQ(keycode_from_str("^I"), MOD_CTRL | 'i');
1293 1 EXPECT_EQ(keycode_from_str("^M"), MOD_CTRL | 'm');
1294 1 EXPECT_EQ(keycode_from_str("C-I"), MOD_CTRL | 'i');
1295 1 EXPECT_EQ(keycode_from_str("C-M"), MOD_CTRL | 'm');
1296 1 EXPECT_EQ(keycode_from_str("C-F1"), MOD_CTRL | KEY_F1);
1297 1 EXPECT_EQ(keycode_from_str("C-M-S-F20"), MOD_CTRL | MOD_META | MOD_SHIFT | KEY_F20);
1298 1 EXPECT_EQ(keycode_from_str("C-M-S-F24"), MOD_CTRL | MOD_META | MOD_SHIFT | KEY_F24);
1299 1 EXPECT_EQ(keycode_from_str("C-M-S-F35"), MOD_CTRL | MOD_META | MOD_SHIFT | KEY_F35);
1300 1 EXPECT_EQ(keycode_from_str("s-m"), MOD_SUPER | 'm');
1301 1 EXPECT_EQ(keycode_from_str("H-y"), MOD_HYPER | 'y');
1302 1 EXPECT_EQ(keycode_from_str("H-y"), MOD_HYPER | 'y');
1303 1 EXPECT_EQ(keycode_from_str("H-S-z"), MOD_HYPER | MOD_SHIFT | 'z');
1304 1 EXPECT_EQ(keycode_from_str("s-S-t"), MOD_SUPER | MOD_SHIFT | 't');
1305 1 EXPECT_EQ(keycode_from_str("S-print"), MOD_SHIFT | KEY_PRINT_SCREEN);
1306 1 EXPECT_EQ(keycode_from_str("C-S-scrlock"), MOD_CTRL | MOD_SHIFT | KEY_SCROLL_LOCK);
1307 1 EXPECT_EQ(keycode_from_str("tab"), KEY_TAB);
1308 1 EXPECT_EQ(keycode_from_str("enter"), KEY_ENTER);
1309 1 EXPECT_EQ(keycode_from_str("space"), KEY_SPACE);
1310 1 EXPECT_EQ(keycode_from_str("escape"), KEY_ESCAPE);
1311 1 EXPECT_EQ(keycode_from_str("backspace"), KEY_BACKSPACE);
1312 1 EXPECT_EQ(keycode_from_str("insert"), KEY_INSERT);
1313 1 EXPECT_EQ(keycode_from_str("delete"), KEY_DELETE);
1314 1 EXPECT_EQ(keycode_from_str("up"), KEY_UP);
1315 1 EXPECT_EQ(keycode_from_str("down"), KEY_DOWN);
1316 1 EXPECT_EQ(keycode_from_str("right"), KEY_RIGHT);
1317 1 EXPECT_EQ(keycode_from_str("left"), KEY_LEFT);
1318 1 EXPECT_EQ(keycode_from_str("begin"), KEY_BEGIN);
1319 1 EXPECT_EQ(keycode_from_str("end"), KEY_END);
1320 1 EXPECT_EQ(keycode_from_str("pgdown"), KEY_PAGE_DOWN);
1321 1 EXPECT_EQ(keycode_from_str("home"), KEY_HOME);
1322 1 EXPECT_EQ(keycode_from_str("pgup"), KEY_PAGE_UP);
1323 1 EXPECT_EQ(keycode_from_str("scrlock"), KEY_SCROLL_LOCK);
1324 1 EXPECT_EQ(keycode_from_str("print"), KEY_PRINT_SCREEN);
1325 1 EXPECT_EQ(keycode_from_str("pause"), KEY_PAUSE);
1326 1 EXPECT_EQ(keycode_from_str("menu"), KEY_MENU);
1327 1 EXPECT_EQ(keycode_from_str("f1"), KEY_F1);
1328 1 EXPECT_EQ(keycode_from_str("f2"), KEY_F2);
1329 1 EXPECT_EQ(keycode_from_str("f3"), KEY_F3);
1330 1 EXPECT_EQ(keycode_from_str("f4"), KEY_F4);
1331 1 EXPECT_EQ(keycode_from_str("f5"), KEY_F5);
1332 1 EXPECT_EQ(keycode_from_str("f6"), KEY_F6);
1333 1 EXPECT_EQ(keycode_from_str("f7"), KEY_F7);
1334 1 EXPECT_EQ(keycode_from_str("f8"), KEY_F8);
1335 1 EXPECT_EQ(keycode_from_str("f9"), KEY_F9);
1336 1 EXPECT_EQ(keycode_from_str("f10"), KEY_F10);
1337 1 EXPECT_EQ(keycode_from_str("f11"), KEY_F11);
1338 1 EXPECT_EQ(keycode_from_str("f12"), KEY_F12);
1339 1 EXPECT_EQ(keycode_from_str("f13"), KEY_F13);
1340 1 EXPECT_EQ(keycode_from_str("f14"), KEY_F14);
1341 1 EXPECT_EQ(keycode_from_str("f15"), KEY_F15);
1342 1 EXPECT_EQ(keycode_from_str("f16"), KEY_F16);
1343 1 EXPECT_EQ(keycode_from_str("f17"), KEY_F17);
1344 1 EXPECT_EQ(keycode_from_str("f18"), KEY_F18);
1345 1 EXPECT_EQ(keycode_from_str("f19"), KEY_F19);
1346 1 EXPECT_EQ(keycode_from_str("f20"), KEY_F20);
1347 1 EXPECT_EQ(keycode_from_str("f21"), KEY_F21);
1348 1 EXPECT_EQ(keycode_from_str("f22"), KEY_F22);
1349 1 EXPECT_EQ(keycode_from_str("f23"), KEY_F23);
1350 1 EXPECT_EQ(keycode_from_str("f24"), KEY_F24);
1351 1 EXPECT_EQ(keycode_from_str("f25"), KEY_F25);
1352 1 EXPECT_EQ(keycode_from_str("f26"), KEY_F26);
1353 1 EXPECT_EQ(keycode_from_str("f27"), KEY_F27);
1354 1 EXPECT_EQ(keycode_from_str("f28"), KEY_F28);
1355 1 EXPECT_EQ(keycode_from_str("f29"), KEY_F29);
1356 1 EXPECT_EQ(keycode_from_str("f30"), KEY_F30);
1357 1 EXPECT_EQ(keycode_from_str("f31"), KEY_F31);
1358 1 EXPECT_EQ(keycode_from_str("f32"), KEY_F32);
1359 1 EXPECT_EQ(keycode_from_str("f33"), KEY_F33);
1360 1 EXPECT_EQ(keycode_from_str("f34"), KEY_F34);
1361 1 EXPECT_EQ(keycode_from_str("f35"), KEY_F35);
1362
1363 1 EXPECT_EQ(keycode_from_str("C-"), KEY_NONE);
1364 1 EXPECT_EQ(keycode_from_str("C-M-"), KEY_NONE);
1365 1 EXPECT_EQ(keycode_from_str("paste"), KEY_NONE);
1366 1 EXPECT_EQ(keycode_from_str("???"), KEY_NONE);
1367 1 EXPECT_EQ(keycode_from_str("F0"), KEY_NONE);
1368 1 EXPECT_EQ(keycode_from_str("F36"), KEY_NONE);
1369 1 EXPECT_EQ(keycode_from_str("F01"), KEY_NONE);
1370 1 EXPECT_EQ(keycode_from_str("\t"), KEY_NONE);
1371 1 EXPECT_EQ(keycode_from_str("\n"), KEY_NONE);
1372 1 EXPECT_EQ(keycode_from_str("\r"), KEY_NONE);
1373 1 EXPECT_EQ(keycode_from_str("\x1f"), KEY_NONE);
1374 1 EXPECT_EQ(keycode_from_str("\x7f"), KEY_NONE);
1375 1 EXPECT_EQ(keycode_from_str("C-\t"), KEY_NONE);
1376 1 EXPECT_EQ(keycode_from_str("C-\r"), KEY_NONE);
1377 1 EXPECT_EQ(keycode_from_str("C-\x7f"), KEY_NONE);
1378
1379 // Special cases for normalization:
1380 1 EXPECT_EQ(keycode_from_str("C-A"), MOD_CTRL | 'a');
1381 1 EXPECT_EQ(keycode_from_str("C-S-A"), MOD_CTRL | MOD_SHIFT | 'a');
1382 1 EXPECT_EQ(keycode_from_str("M-A"), MOD_META | MOD_SHIFT | 'a');
1383 1 EXPECT_EQ(keycode_from_str("C-?"), MOD_CTRL | '?');
1384 1 EXPECT_EQ(keycode_from_str("C-H"), MOD_CTRL | 'h');
1385 1 EXPECT_EQ(keycode_from_str("M-C-?"), MOD_META | MOD_CTRL | '?');
1386 1 EXPECT_EQ(keycode_from_str("M-C-H"), MOD_META | MOD_CTRL | 'h');
1387 1 }
1388
1389 26 static bool clear_obuf(TermOutputBuffer *obuf)
1390 {
1391
1/2
✓ Branch 2 → 3 taken 26 times.
✗ Branch 2 → 4 not taken.
26 if (unlikely(obuf_avail(obuf) <= 8)) {
1392 return false;
1393 }
1394 26 memset(obuf->buf, '\0', obuf->count + 8);
1395 26 obuf->count = 0;
1396 26 obuf->x = 0;
1397 26 return true;
1398 }
1399
1400 1 static void test_term_init(TestContext *ctx)
1401 {
1402 1 const TermFeatureFlags expected_xterm_flags =
1403 TFLAG_256_COLOR |
1404 TFLAG_16_COLOR |
1405 TFLAG_8_COLOR |
1406 TFLAG_BACK_COLOR_ERASE |
1407 TFLAG_SET_WINDOW_TITLE |
1408 TFLAG_OSC52_COPY
1409 ;
1410
1411 1 Terminal term = {.obuf = TERM_OUTPUT_INIT};
1412 1 term_init(&term, "xterm-256color", NULL);
1413 1 EXPECT_EQ(term.width, 80);
1414 1 EXPECT_EQ(term.height, 24);
1415 1 EXPECT_EQ(term.ncv_attributes, 0);
1416 1 EXPECT_EQ(term.features, expected_xterm_flags);
1417
1418 1 term_init(&term, "ansi", NULL);
1419 1 EXPECT_EQ(term.width, 80);
1420 1 EXPECT_EQ(term.height, 24);
1421 1 EXPECT_EQ(term.ncv_attributes, ATTR_UNDERLINE);
1422 1 EXPECT_EQ(term.features, TFLAG_8_COLOR | TFLAG_NCV_UNDERLINE);
1423
1424 1 term_init(&term, "ansi-m", NULL);
1425 1 EXPECT_EQ(term.width, 80);
1426 1 EXPECT_EQ(term.height, 24);
1427 1 EXPECT_EQ(term.ncv_attributes, 0);
1428 1 EXPECT_EQ(term.features, 0);
1429 1 }
1430
1431 1 static void test_term_put_str(TestContext *ctx)
1432 {
1433 1 Terminal term = {
1434 .width = 80,
1435 .height = 24,
1436 .obuf = TERM_OUTPUT_INIT,
1437 };
1438
1439 // Fill start of buffer with zeroes, to allow using EXPECT_STREQ() below
1440 1 TermOutputBuffer *obuf = &term.obuf;
1441 1 ASSERT_TRUE(256 <= TERM_OUTBUF_SIZE);
1442 1 memset(obuf->buf, 0, 256);
1443
1444 1 obuf->width = 0;
1445 1 term_put_str(obuf, "this should write nothing because obuf->width == 0");
1446 1 EXPECT_EQ(obuf->count, 0);
1447 1 EXPECT_EQ(obuf->x, 0);
1448
1449 1 term_output_reset(&term, 0, 80, 0);
1450 1 EXPECT_EQ(obuf->tab_mode, TAB_CONTROL);
1451 1 EXPECT_EQ(obuf->tab_width, 8);
1452 1 EXPECT_EQ(obuf->x, 0);
1453 1 EXPECT_EQ(obuf->width, 80);
1454 1 EXPECT_EQ(obuf->scroll_x, 0);
1455 1 EXPECT_EQ(term.width, 80);
1456 1 EXPECT_EQ(obuf->can_clear, true);
1457
1458 1 term_put_str(obuf, "1\xF0\x9F\xA7\xB2 \t xyz \t\r \xC2\xB6");
1459 1 EXPECT_EQ(obuf->count, 20);
1460 1 EXPECT_EQ(obuf->x, 17);
1461 1 EXPECT_STREQ(obuf->buf, "1\xF0\x9F\xA7\xB2 ^I xyz ^I^M \xC2\xB6");
1462
1463 1 EXPECT_TRUE(term_put_char(obuf, 0x10FFFF));
1464 1 EXPECT_EQ(obuf->count, 24);
1465 1 EXPECT_EQ(obuf->x, 21);
1466 1 EXPECT_STREQ(obuf->buf + 20, "<" "??" ">");
1467 1 ASSERT_TRUE(clear_obuf(obuf));
1468 1 }
1469
1470 1 static void test_term_clear_eol(TestContext *ctx)
1471 {
1472 1 Terminal term = {
1473 .width = 80,
1474 .height = 24,
1475 .obuf = TERM_OUTPUT_INIT,
1476 };
1477
1478 // BCE with non-default bg
1479 1 TermOutputBuffer *obuf = &term.obuf;
1480 1 term.features = TFLAG_BACK_COLOR_ERASE;
1481 1 obuf->style = (TermStyle){.bg = COLOR_RED};
1482 1 term_output_reset(&term, 0, 80, 0);
1483 1 EXPECT_EQ(term_clear_eol(&term), TERM_CLEAR_EOL_USED_EL);
1484 1 EXPECT_MEMEQ(obuf->buf, obuf->count, "\033[K", 3);
1485 1 EXPECT_EQ(obuf->x, 80);
1486 1 ASSERT_TRUE(clear_obuf(obuf));
1487
1488 // No BCE, but with REP
1489 1 term.features = TFLAG_ECMA48_REPEAT;
1490 1 obuf->style = (TermStyle){.bg = COLOR_RED};
1491 1 term_output_reset(&term, 0, 40, 0);
1492 1 EXPECT_EQ(term_clear_eol(&term), -40);
1493 1 EXPECT_MEMEQ(obuf->buf, obuf->count, " \033[39b", 6);
1494 1 EXPECT_EQ(obuf->x, 40);
1495 1 ASSERT_TRUE(clear_obuf(obuf));
1496
1497 // No BCE with non-default bg
1498 1 term.features = 0;
1499 1 obuf->style = (TermStyle){.bg = COLOR_RED};
1500 1 term_output_reset(&term, 0, 80, 0);
1501 1 EXPECT_EQ(term_clear_eol(&term), 80);
1502 1 EXPECT_EQ(obuf->count, 80);
1503 1 EXPECT_EQ(obuf->x, 80);
1504 1 EXPECT_EQ(obuf->buf[0], ' ');
1505 1 EXPECT_EQ(obuf->buf[79], ' ');
1506 1 ASSERT_TRUE(clear_obuf(obuf));
1507
1508 // No BCE with default bg/attrs
1509 1 term.features = 0;
1510 1 obuf->style = (TermStyle){.bg = COLOR_DEFAULT};
1511 1 term_output_reset(&term, 0, 80, 0);
1512 1 EXPECT_EQ(term_clear_eol(&term), TERM_CLEAR_EOL_USED_EL);
1513 1 EXPECT_MEMEQ(obuf->buf, obuf->count, "\033[K", 3);
1514 1 EXPECT_EQ(obuf->x, 80);
1515 1 ASSERT_TRUE(clear_obuf(obuf));
1516
1517 // No BCE with ATTR_REVERSE
1518 1 term.features = 0;
1519 1 obuf->style = (TermStyle){.bg = COLOR_DEFAULT, .attr = ATTR_REVERSE};
1520 1 term_output_reset(&term, 0, 80, 0);
1521 1 EXPECT_EQ(term_clear_eol(&term), 80);
1522 1 EXPECT_EQ(obuf->count, 80);
1523 1 EXPECT_EQ(obuf->x, 80);
1524 1 ASSERT_TRUE(clear_obuf(obuf));
1525
1526 // x >= scroll_x + width
1527 1 term_output_reset(&term, 0, 20, 10);
1528 1 EXPECT_EQ(obuf->width, 20);
1529 1 EXPECT_EQ(obuf->scroll_x, 10);
1530 1 obuf->x = 30;
1531 1 EXPECT_EQ(term_clear_eol(&term), 0);
1532 1 EXPECT_EQ(obuf->count, 0);
1533 1 EXPECT_EQ(obuf->x, 30);
1534 1 ASSERT_TRUE(clear_obuf(obuf));
1535 1 }
1536
1537 1 static void test_term_move_cursor(TestContext *ctx)
1538 {
1539 1 TermOutputBuffer obuf = TERM_OUTPUT_INIT;
1540 1 term_move_cursor(&obuf, 12, 5);
1541 1 EXPECT_MEMEQ(obuf.buf, obuf.count, "\033[6;13H", 7);
1542 1 EXPECT_EQ(obuf.x, 0);
1543 1 ASSERT_TRUE(clear_obuf(&obuf));
1544
1545 1 term_move_cursor(&obuf, 0, 22);
1546 1 EXPECT_MEMEQ(obuf.buf, obuf.count, "\033[23H", 5);
1547 1 EXPECT_EQ(obuf.x, 0);
1548 1 ASSERT_TRUE(clear_obuf(&obuf));
1549 1 }
1550
1551 1 static void test_term_set_bytes(TestContext *ctx)
1552 {
1553 1 Terminal term = {
1554 .width = 80,
1555 .height = 24,
1556 .obuf = TERM_OUTPUT_INIT,
1557 .features = TFLAG_ECMA48_REPEAT,
1558 };
1559
1560 1 TermOutputBuffer *obuf = &term.obuf;
1561 1 ASSERT_TRUE(clear_obuf(obuf));
1562 1 term_output_reset(&term, 0, 80, 0);
1563
1564 1 EXPECT_EQ(term_set_bytes(&term, 'x', 40), TERM_SET_BYTES_REP);
1565 1 EXPECT_MEMEQ(obuf->buf, obuf->count, "x\033[39b", 6);
1566 1 EXPECT_EQ(obuf->x, 40);
1567 1 ASSERT_TRUE(clear_obuf(obuf));
1568
1569 1 const size_t repmin = ECMA48_REP_MIN - 1;
1570 1 EXPECT_EQ(repmin, 5);
1571 1 EXPECT_EQ(term_set_bytes(&term, '-', repmin), TERM_SET_BYTES_MEMSET);
1572 1 EXPECT_MEMEQ(obuf->buf, obuf->count, "-----", repmin);
1573 1 EXPECT_EQ(obuf->x, repmin);
1574 1 ASSERT_TRUE(clear_obuf(obuf));
1575
1576 1 EXPECT_EQ(term_set_bytes(&term, '\n', 8), TERM_SET_BYTES_MEMSET);
1577 1 EXPECT_MEMEQ(obuf->buf, obuf->count, "\n\n\n\n\n\n\n\n", 8);
1578 1 EXPECT_EQ(obuf->x, 8);
1579 1 ASSERT_TRUE(clear_obuf(obuf));
1580 1 }
1581
1582 1 static void test_term_set_style(TestContext *ctx)
1583 {
1584 1 Terminal term = {.obuf = TERM_OUTPUT_INIT};
1585 1 term_init(&term, "tmux", "truecolor");
1586 1 EXPECT_TRUE(term.features & TFLAG_TRUE_COLOR);
1587 1 EXPECT_EQ(term.ncv_attributes, 0);
1588
1589 1 TermOutputBuffer *obuf = &term.obuf;
1590 1 ASSERT_TRUE(TERM_OUTBUF_SIZE >= 64);
1591 1 memset(obuf->buf, '?', 64);
1592
1593 1 TermStyle style = {
1594 .fg = COLOR_RED,
1595 .bg = COLOR_YELLOW,
1596 .attr = ATTR_BOLD | ATTR_REVERSE,
1597 };
1598
1599 1 term_set_style(&term, style);
1600 1 EXPECT_MEMEQ(obuf->buf, obuf->count, "\033[0;1;7;31;43m", 14);
1601 1 EXPECT_EQ(obuf->x, 0);
1602 1 ASSERT_TRUE(clear_obuf(obuf));
1603
1604 1 style.attr = 0;
1605 1 style.fg = COLOR_RGB(0x12ef46);
1606 1 term_set_style(&term, style);
1607 1 EXPECT_MEMEQ(obuf->buf, obuf->count, "\033[0;38;2;18;239;70;43m", 22);
1608 1 EXPECT_EQ(obuf->x, 0);
1609 1 ASSERT_TRUE(clear_obuf(obuf));
1610
1611 1 style.fg = 144;
1612 1 style.bg = COLOR_DEFAULT;
1613 1 term_set_style(&term, style);
1614 1 EXPECT_MEMEQ(obuf->buf, obuf->count, "\033[0;38;5;144m", 13);
1615 1 EXPECT_EQ(obuf->x, 0);
1616 1 ASSERT_TRUE(clear_obuf(obuf));
1617
1618 1 style.fg = COLOR_GRAY;
1619 1 style.bg = COLOR_RGB(0x00b91f);
1620 1 term_set_style(&term, style);
1621 1 EXPECT_MEMEQ(obuf->buf, obuf->count, "\033[0;37;48;2;0;185;31m", 21);
1622 1 EXPECT_EQ(obuf->x, 0);
1623 1 ASSERT_TRUE(clear_obuf(obuf));
1624
1625 1 style.fg = COLOR_WHITE;
1626 1 style.bg = COLOR_DARKGRAY;
1627 1 style.attr = ATTR_ITALIC;
1628 1 term_set_style(&term, style);
1629 1 EXPECT_MEMEQ(obuf->buf, obuf->count, "\033[0;3;97;100m", 13);
1630 1 EXPECT_EQ(obuf->x, 0);
1631 1 ASSERT_TRUE(clear_obuf(obuf));
1632
1633 // Ensure longest sequence doesn't trigger assertion
1634 1 static const char longest[] =
1635 "\033[0;"
1636 "1;2;3;4;5;7;8;9;"
1637 "38;2;100;101;199;"
1638 "48;2;200;202;255m"
1639 ;
1640 1 const size_t n = sizeof(longest) - 1;
1641 1 ASSERT_EQ(n, 54);
1642 1 style.fg = COLOR_RGB(0x6465C7);
1643 1 style.bg = COLOR_RGB(0xC8CAFF);
1644 1 style.attr = ATTR_ALL;
1645 1 term_set_style(&term, style);
1646 1 EXPECT_MEMEQ(obuf->buf, obuf->count, longest, n);
1647 1 EXPECT_EQ(obuf->x, 0);
1648 1 ASSERT_TRUE(clear_obuf(obuf));
1649
1650 1 style.fg = COLOR_DEFAULT;
1651 1 style.bg = COLOR_DEFAULT;
1652 1 style.attr = ATTR_REVERSE | ATTR_DIM | ATTR_UNDERLINE | ATTR_KEEP;
1653 1 term.ncv_attributes = ATTR_DIM | ATTR_STRIKETHROUGH;
1654 1 term_set_style(&term, style);
1655 1 EXPECT_MEMEQ(obuf->buf, obuf->count, "\033[0;2;4;7m", 10);
1656 1 EXPECT_EQ(obuf->x, 0);
1657 1 style.attr &= ~ATTR_KEEP;
1658 1 EXPECT_TRUE(same_style(&obuf->style, &style));
1659 1 ASSERT_TRUE(clear_obuf(obuf));
1660
1661 1 style.fg = COLOR_BLUE;
1662 1 term_set_style(&term, style);
1663 1 EXPECT_MEMEQ(obuf->buf, obuf->count, "\033[0;4;7;34m", 11);
1664 1 EXPECT_EQ(obuf->x, 0);
1665 1 style.attr &= ~ATTR_DIM;
1666 1 EXPECT_TRUE(same_style(&obuf->style, &style));
1667 1 ASSERT_TRUE(clear_obuf(obuf));
1668 1 }
1669
1670 1 static void test_term_osc52_copy(TestContext *ctx)
1671 {
1672 1 TermOutputBuffer obuf = TERM_OUTPUT_INIT;
1673 1 StringView text = strview("foobar");
1674 1 EXPECT_TRUE(term_osc52_copy(&obuf, text, TCOPY_CLIPBOARD | TCOPY_PRIMARY));
1675 1 EXPECT_MEMEQ(obuf.buf, obuf.count, "\033]52;pc;Zm9vYmFy\033\\", 18);
1676 1 EXPECT_EQ(obuf.x, 0);
1677 1 ASSERT_TRUE(clear_obuf(&obuf));
1678
1679 1 text = strview("\xF0\x9F\xA5\xA3");
1680 1 EXPECT_TRUE(term_osc52_copy(&obuf, text, TCOPY_PRIMARY));
1681 1 EXPECT_MEMEQ(obuf.buf, obuf.count, "\033]52;p;8J+low==\033\\", 17);
1682 1 EXPECT_EQ(obuf.x, 0);
1683 1 ASSERT_TRUE(clear_obuf(&obuf));
1684
1685 1 text = strview("");
1686 1 EXPECT_TRUE(term_osc52_copy(&obuf, text, TCOPY_CLIPBOARD));
1687 1 EXPECT_MEMEQ(obuf.buf, obuf.count, "\033]52;c;\033\\", 9);
1688 1 EXPECT_EQ(obuf.x, 0);
1689 1 ASSERT_TRUE(clear_obuf(&obuf));
1690 1 }
1691
1692 1 static void test_term_set_cursor_style(TestContext *ctx)
1693 {
1694 1 Terminal term = {
1695 .width = 80,
1696 .height = 24,
1697 .obuf = TERM_OUTPUT_INIT,
1698 };
1699
1700 1 TermCursorStyle style = {
1701 .type = CURSOR_STEADY_BAR,
1702 .color = COLOR_RGB(0x22AACC),
1703 };
1704
1705 1 static const char expected[] = "\033[6 q\033]12;rgb:22/aa/cc\033\\";
1706 1 size_t expected_len = sizeof(expected) - 1;
1707 1 ASSERT_EQ(expected_len, 24);
1708
1709 1 TermOutputBuffer *obuf = &term.obuf;
1710 1 memset(obuf->buf, '@', expected_len + 16);
1711
1712 1 term_set_cursor_style(&term, style);
1713 1 EXPECT_MEMEQ(obuf->buf, obuf->count, expected, expected_len);
1714 1 EXPECT_EQ(obuf->x, 0);
1715 1 EXPECT_EQ(obuf->cursor_style.type, style.type);
1716 1 EXPECT_EQ(obuf->cursor_style.color, style.color);
1717 1 ASSERT_TRUE(clear_obuf(obuf));
1718 1 }
1719
1720 1 static void test_term_restore_cursor_style(TestContext *ctx)
1721 {
1722 1 Terminal term = {
1723 .width = 80,
1724 .height = 24,
1725 .obuf = TERM_OUTPUT_INIT,
1726 };
1727
1728 1 static const char expected[] = "\033[0 q\033]112\033\\";
1729 1 size_t expected_len = sizeof(expected) - 1;
1730 1 ASSERT_EQ(expected_len, 12);
1731
1732 1 TermOutputBuffer *obuf = &term.obuf;
1733 1 memset(obuf->buf, '@', expected_len + 16);
1734
1735 1 term_restore_cursor_style(&term);
1736 1 EXPECT_MEMEQ(obuf->buf, obuf->count, expected, expected_len);
1737 1 EXPECT_EQ(obuf->x, 0);
1738 1 ASSERT_TRUE(clear_obuf(obuf));
1739 1 }
1740
1741 1 static void test_term_begin_sync_update(TestContext *ctx)
1742 {
1743 1 Terminal term = {.obuf = TERM_OUTPUT_INIT};
1744 1 term_init(&term, "xterm-kitty", NULL);
1745 1 EXPECT_TRUE(term.features & TFLAG_SYNC);
1746
1747 1 static const char expected[] =
1748 "\033[?2026h"
1749 "\033[?1049h"
1750 "\033[?25l"
1751 "\033[?25h"
1752 "\033[?1049l"
1753 "\033[?2026l"
1754 ;
1755
1756 1 TermOutputBuffer *obuf = &term.obuf;
1757 1 memset(obuf->buf, '.', 128);
1758
1759 1 term_begin_sync_update(&term);
1760 1 term_use_alt_screen_buffer(&term);
1761 1 term_hide_cursor(&term);
1762 1 term_show_cursor(&term);
1763 1 term_use_normal_screen_buffer(&term);
1764 1 term_end_sync_update(&term);
1765
1766 1 EXPECT_MEMEQ(obuf->buf, obuf->count, expected, sizeof(expected) - 1);
1767 1 EXPECT_EQ(obuf->x, 0);
1768 1 }
1769
1770 1 static void test_term_put_level_1_queries(TestContext *ctx)
1771 {
1772 1 enum {
1773 // Short aliases for TermFeatureFlags:
1774 BCE = TFLAG_BACK_COLOR_ERASE,
1775 TITLE = TFLAG_SET_WINDOW_TITLE,
1776 OSC52 = TFLAG_OSC52_COPY,
1777 C8 = TFLAG_8_COLOR,
1778 C16 = TFLAG_16_COLOR | C8,
1779 C256 = TFLAG_256_COLOR | C16,
1780 };
1781
1782 1 Terminal term = {.obuf = TERM_OUTPUT_INIT};
1783 1 TermOutputBuffer *obuf = &term.obuf;
1784 1 term_init(&term, "xterm-256color", NULL);
1785 1 EXPECT_UINT_EQ(term.features, (C256 | BCE | TITLE | OSC52));
1786 1 EXPECT_EQ(obuf->count, 0);
1787
1788 // Basic level 1 queries
1789 1 static const char level1[] = "\033[c";
1790 1 term_put_initial_queries(&term, 1);
1791 1 EXPECT_MEMEQ(obuf->buf, obuf->count, level1, sizeof(level1) - 1);
1792
1793 // All queries (forced by emit_all=true)
1794 1 static const char full[] =
1795 // term_put_initial_queries()
1796 "\033[c"
1797 // term_put_level_2_queries()
1798 "\033[>0q"
1799 "\033[>c"
1800 "\033[?u"
1801 "\033[?1036$p"
1802 "\033[?1039$p"
1803 "\033[?2026$p"
1804 // term_put_level_2_queries() debug
1805 "\033[?4m"
1806 "\033[?7$p"
1807 "\033[?25$p"
1808 "\033[?45$p"
1809 "\033[?67$p"
1810 "\033[?1049$p"
1811 "\033[?2004$p"
1812 "\033[18t"
1813 // term_put_level_3_queries()
1814 "\033[0;38;2;60;70;80;48;5;255m"
1815 "\033P$qm\033\\"
1816 "\033[0m"
1817 "\033P+q626365\033\\"
1818 "\033P+q726570\033\\"
1819 "\033P+q74736C\033\\"
1820 "\033P+q4D73\033\\"
1821 // term_put_level_3_queries() debug
1822 "\033P+q71756572792d6f732d6e616d65\033\\"
1823 "\033P$q q\033\\"
1824 ;
1825
1826 1 obuf->count = 0;
1827 1 term_put_initial_queries(&term, 6);
1828 1 EXPECT_MEMEQ(obuf->buf, obuf->count, full, sizeof(full) - 1);
1829
1830 1 EXPECT_EQ(obuf->scroll_x, 0);
1831 1 EXPECT_EQ(obuf->x, 0);
1832 1 }
1833
1834 1 static void test_update_term_title(TestContext *ctx)
1835 {
1836 1 static const char prefix[] = "\033]2;";
1837 1 static const char suffix[] = " - dte\033\\";
1838 1 size_t plen = sizeof(prefix) - 1;
1839 1 size_t slen = sizeof(suffix) - 1;
1840 1 TermOutputBuffer obuf = TERM_OUTPUT_INIT;
1841
1842 1 static const char expected1[] = "\033]2;example filename - dte\033\\";
1843 1 update_term_title(&obuf, strview("example filename"), false);
1844 1 EXPECT_MEMEQ(obuf.buf, obuf.count, expected1, sizeof(expected1) - 1);
1845 1 EXPECT_EQ(obuf.x, 0);
1846 1 obuf.count = 0;
1847
1848 // Control char escaping
1849 1 static const char expected2[] = "\033]2;x^I^H^[y - dte\033\\";
1850 1 update_term_title(&obuf, strview("x\t\b\033y"), false);
1851 1 EXPECT_MEMEQ(obuf.buf, obuf.count, expected2, sizeof(expected2) - 1);
1852 1 EXPECT_EQ(obuf.x, 0);
1853 1 obuf.count = 0;
1854
1855 // Very long filename, for edge case coverage
1856 1 char filename[TERM_OUTBUF_SIZE];
1857 1 memset(filename, 'a', sizeof(filename));
1858 1 filename[sizeof(filename) - 1] = '\0';
1859 1 update_term_title(&obuf, strview(filename), false);
1860
1861 1 size_t tlen = obuf.count - (plen + slen);
1862 1 EXPECT_EQ(sizeof(filename), 8192);
1863 1 EXPECT_EQ(obuf.count, 8188);
1864 1 EXPECT_EQ(tlen, 8176);
1865 1 EXPECT_MEMEQ(obuf.buf, plen, prefix, plen);
1866 1 EXPECT_MEMEQ(obuf.buf + plen, tlen, filename, tlen);
1867 1 EXPECT_MEMEQ(obuf.buf + plen + tlen, slen, suffix, slen);
1868 1 }
1869
1870 static const TestEntry tests[] = {
1871 TEST(test_parse_rgb),
1872 TEST(test_parse_term_style),
1873 TEST(test_color_to_nearest),
1874 TEST(test_color_to_str),
1875 TEST(test_term_style_to_string),
1876 TEST(test_cursor_mode_from_str),
1877 TEST(test_cursor_type_from_str),
1878 TEST(test_cursor_color_from_str),
1879 TEST(test_cursor_color_to_str),
1880 TEST(test_same_cursor),
1881 TEST(test_mask_style),
1882 TEST(test_term_parse_csi_params),
1883 TEST(test_term_parse_sequence),
1884 TEST(test_term_parse_sequence2),
1885 TEST(test_rxvt_parse_key),
1886 TEST(test_linux_parse_key),
1887 TEST(test_keycode_to_string),
1888 TEST(test_parse_key_string),
1889 TEST(test_term_init),
1890 TEST(test_term_put_str),
1891 TEST(test_term_clear_eol),
1892 TEST(test_term_move_cursor),
1893 TEST(test_term_set_bytes),
1894 TEST(test_term_set_style),
1895 TEST(test_term_osc52_copy),
1896 TEST(test_term_set_cursor_style),
1897 TEST(test_term_restore_cursor_style),
1898 TEST(test_term_begin_sync_update),
1899 TEST(test_term_put_level_1_queries),
1900 TEST(test_update_term_title),
1901 };
1902
1903 const TestGroup terminal_tests = TEST_GROUP(tests);
1904