2016-11-14 17:40:01 +03:00
|
|
|
#!/usr/bin/env python
|
|
|
|
# vim:fileencoding=utf-8
|
|
|
|
# License: GPL v3 Copyright: 2016, Kovid Goyal <kovid at kovidgoyal.net>
|
|
|
|
|
2016-11-16 06:37:15 +03:00
|
|
|
from functools import partial
|
|
|
|
|
2016-11-14 17:40:01 +03:00
|
|
|
from . import BaseTest
|
|
|
|
|
2016-11-17 11:20:20 +03:00
|
|
|
from kitty.fast_data_types import parse_bytes, parse_bytes_dump, CURSOR_BLOCK
|
2016-11-14 18:10:54 +03:00
|
|
|
|
|
|
|
|
|
|
|
class CmdDump(list):
|
|
|
|
|
|
|
|
def __call__(self, *a):
|
|
|
|
self.append(a)
|
2016-11-14 17:40:01 +03:00
|
|
|
|
|
|
|
|
2016-11-23 13:03:08 +03:00
|
|
|
class TestParser(BaseTest):
|
2016-11-14 17:40:01 +03:00
|
|
|
|
2016-11-17 11:20:20 +03:00
|
|
|
def parse_bytes_dump(self, s, x, *cmds):
|
2016-11-16 06:37:15 +03:00
|
|
|
cd = CmdDump()
|
|
|
|
if isinstance(x, str):
|
|
|
|
x = x.encode('utf-8')
|
2016-11-23 13:17:22 +03:00
|
|
|
cmds = tuple(('draw', x) if isinstance(x, str) else x for x in cmds)
|
2016-11-18 13:00:59 +03:00
|
|
|
parse_bytes_dump(cd, s, x)
|
2016-11-23 13:17:22 +03:00
|
|
|
current = ''
|
|
|
|
q = []
|
|
|
|
for args in cd:
|
|
|
|
if args[0] == 'draw':
|
2016-11-24 11:13:28 +03:00
|
|
|
if args[1] is not None:
|
|
|
|
current += args[1]
|
2016-11-23 13:17:22 +03:00
|
|
|
else:
|
|
|
|
if current:
|
|
|
|
q.append(('draw', current))
|
|
|
|
current = ''
|
|
|
|
q.append(args)
|
|
|
|
if current:
|
|
|
|
q.append(('draw', current))
|
|
|
|
self.ae(tuple(q), cmds)
|
2016-11-16 06:37:15 +03:00
|
|
|
|
2016-11-14 17:40:01 +03:00
|
|
|
def test_simple_parsing(self):
|
|
|
|
s = self.create_screen()
|
2016-11-17 11:20:20 +03:00
|
|
|
pb = partial(self.parse_bytes_dump, s)
|
2016-11-14 18:10:54 +03:00
|
|
|
|
2016-11-18 13:59:15 +03:00
|
|
|
pb('12', '12')
|
2016-11-14 17:40:01 +03:00
|
|
|
self.ae(str(s.line(0)), '12 ')
|
2016-11-18 14:16:04 +03:00
|
|
|
self.ae(s.cursor.x, 2)
|
2016-11-18 13:59:15 +03:00
|
|
|
pb('3456', '3456')
|
2016-11-14 17:40:01 +03:00
|
|
|
self.ae(str(s.line(0)), '12345')
|
|
|
|
self.ae(str(s.line(1)), '6 ')
|
2016-11-18 13:59:15 +03:00
|
|
|
pb(b'\n123\n\r45', ('screen_linefeed', ord('\n')), '123', ('screen_linefeed', ord('\n')), ('screen_carriage_return', ord('\r')), '45')
|
2016-11-14 17:40:01 +03:00
|
|
|
self.ae(str(s.line(1)), '6 ')
|
|
|
|
self.ae(str(s.line(2)), ' 123 ')
|
|
|
|
self.ae(str(s.line(3)), '45 ')
|
|
|
|
parse_bytes(s, b'\rabcde')
|
|
|
|
self.ae(str(s.line(3)), 'abcde')
|
2016-11-18 13:59:15 +03:00
|
|
|
pb('\rßxyz1', ('screen_carriage_return', 13), 'ßxyz1')
|
2016-11-14 17:40:01 +03:00
|
|
|
self.ae(str(s.line(3)), 'ßxyz1')
|
2016-11-18 13:59:15 +03:00
|
|
|
pb('ニチ ', 'ニチ ')
|
2016-11-14 17:40:01 +03:00
|
|
|
self.ae(str(s.line(4)), 'ニチ ')
|
2016-11-16 06:37:15 +03:00
|
|
|
|
|
|
|
def test_esc_codes(self):
|
|
|
|
s = self.create_screen()
|
2016-11-17 11:20:20 +03:00
|
|
|
pb = partial(self.parse_bytes_dump, s)
|
2016-11-23 13:17:22 +03:00
|
|
|
pb('12\033Da', '12', ('screen_index', ord('D')), 'a')
|
2016-11-16 06:37:15 +03:00
|
|
|
self.ae(str(s.line(0)), '12 ')
|
|
|
|
self.ae(str(s.line(1)), ' a ')
|
2016-11-23 13:17:22 +03:00
|
|
|
pb('\033x', ('Unknown char after ESC: 0x%x' % ord('x'),))
|
|
|
|
pb('\033c123', ('screen_reset', ord('c')), '123')
|
2016-11-16 06:37:15 +03:00
|
|
|
self.ae(str(s.line(0)), '123 ')
|
2016-11-16 11:10:38 +03:00
|
|
|
|
|
|
|
def test_csi_codes(self):
|
|
|
|
s = self.create_screen()
|
2016-11-17 11:20:20 +03:00
|
|
|
pb = partial(self.parse_bytes_dump, s)
|
2016-11-18 13:59:15 +03:00
|
|
|
pb('abcde', 'abcde')
|
2016-11-16 11:10:38 +03:00
|
|
|
s.cursor_back(5)
|
2016-11-18 13:59:15 +03:00
|
|
|
pb('x\033[2@y', 'x', ('screen_insert_characters', 2), 'y')
|
2016-11-16 11:10:38 +03:00
|
|
|
self.ae(str(s.line(0)), 'xy bc')
|
2016-11-18 13:59:15 +03:00
|
|
|
pb('x\033[2;7@y', 'x', ('screen_insert_characters', 2), 'y')
|
|
|
|
pb('x\033[@y', 'x', ('screen_insert_characters', 1), 'y')
|
|
|
|
pb('x\033[345@y', 'x', ('screen_insert_characters', 345), 'y')
|
|
|
|
pb('x\033[345;@y', 'x', ('screen_insert_characters', 345), 'y')
|
2016-11-16 17:23:38 +03:00
|
|
|
pb('\033[H', ('screen_cursor_position', 1, 1))
|
|
|
|
self.ae(s.cursor.x, 0), self.ae(s.cursor.y, 0)
|
|
|
|
pb('\033[4H', ('screen_cursor_position', 4, 1))
|
|
|
|
pb('\033[3;2H', ('screen_cursor_position', 3, 2))
|
|
|
|
pb('\033[3;2;H', ('screen_cursor_position', 3, 2))
|
|
|
|
self.ae(s.cursor.x, 1), self.ae(s.cursor.y, 2)
|
2016-11-16 17:49:31 +03:00
|
|
|
pb('\033[J', ('screen_erase_in_display', 0, 0))
|
|
|
|
pb('\033[?J', ('screen_erase_in_display', 0, 1))
|
|
|
|
pb('\033[?2J', ('screen_erase_in_display', 2, 1))
|
2016-11-17 06:04:55 +03:00
|
|
|
pb('\033[h')
|
|
|
|
pb('\033[20;4h', ('screen_set_mode', 20), ('screen_set_mode', 4))
|
2016-11-24 18:48:04 +03:00
|
|
|
pb('\033[?1000;1004h', ('screen_set_mode', 1000 << 5), ('screen_set_mode', 1004 << 5))
|
|
|
|
pb('\033[20;4;20l', ('screen_reset_mode', 20), ('screen_reset_mode', 4), ('screen_reset_mode', 20))
|
2016-11-16 11:10:38 +03:00
|
|
|
s.reset()
|
2016-11-17 06:04:55 +03:00
|
|
|
pb('\033[1;3;4;7;9;34;44m', ('select_graphic_rendition', 7))
|
|
|
|
for attr in 'bold italic reverse strikethrough'.split():
|
|
|
|
self.assertTrue(getattr(s.cursor, attr))
|
|
|
|
self.ae(s.cursor.decoration, 1)
|
2016-11-24 12:58:52 +03:00
|
|
|
self.ae(s.cursor.fg, 4 << 8 | 1)
|
|
|
|
self.ae(s.cursor.bg, 4 << 8 | 1)
|
2016-11-17 06:04:55 +03:00
|
|
|
pb('\033[38;5;1;48;5;7m', ('select_graphic_rendition', 6))
|
2016-11-24 12:58:52 +03:00
|
|
|
self.ae(s.cursor.fg, 1 << 8 | 1)
|
|
|
|
self.ae(s.cursor.bg, 7 << 8 | 1)
|
2016-11-17 06:04:55 +03:00
|
|
|
pb('\033[38;2;1;2;3;48;2;7;8;9m', ('select_graphic_rendition', 10))
|
2016-11-24 12:58:52 +03:00
|
|
|
self.ae(s.cursor.fg, 1 << 24 | 2 << 16 | 3 << 8 | 2)
|
|
|
|
self.ae(s.cursor.bg, 7 << 24 | 8 << 16 | 9 << 8 | 2)
|
2016-11-24 13:58:08 +03:00
|
|
|
c = s.callbacks
|
2016-11-17 08:29:15 +03:00
|
|
|
pb('\033[5n', ('report_device_status', 5, 0))
|
|
|
|
self.ae(c.wtcbuf, b'\033[0n')
|
|
|
|
c.clear()
|
|
|
|
pb('\033[6n', ('report_device_status', 6, 0))
|
|
|
|
self.ae(c.wtcbuf, b'\033[1;1R')
|
2016-11-18 13:59:15 +03:00
|
|
|
pb('12345', '12345')
|
2016-11-17 08:29:15 +03:00
|
|
|
c.clear()
|
|
|
|
pb('\033[6n', ('report_device_status', 6, 0))
|
|
|
|
self.ae(c.wtcbuf, b'\033[2;1R')
|
2016-11-17 08:43:28 +03:00
|
|
|
pb('\033[2;4r', ('screen_set_margins', 2, 4))
|
|
|
|
self.ae(s.margin_top, 1), self.ae(s.margin_bottom, 3)
|
|
|
|
pb('\033[r', ('screen_set_margins', 0, 0))
|
|
|
|
self.ae(s.margin_top, 0), self.ae(s.margin_bottom, 4)
|
2016-11-17 11:20:20 +03:00
|
|
|
pb('\033[1 q', ('screen_set_cursor', 1, ord(' ')))
|
|
|
|
self.assertTrue(s.cursor.blink)
|
|
|
|
self.ae(s.cursor.shape, CURSOR_BLOCK)
|
2016-11-18 08:46:17 +03:00
|
|
|
|
|
|
|
def test_osc_codes(self):
|
|
|
|
s = self.create_screen()
|
|
|
|
pb = partial(self.parse_bytes_dump, s)
|
2016-11-24 13:58:08 +03:00
|
|
|
c = s.callbacks
|
2016-11-23 17:52:52 +03:00
|
|
|
pb('a\033]2;xyz\x9cbcde', 'a', ('set_title', 'xyz'), 'bcde')
|
2016-11-18 08:46:17 +03:00
|
|
|
self.ae(str(s.line(0)), 'abcde')
|
2016-11-23 14:55:38 +03:00
|
|
|
self.ae(c.titlebuf, 'xyz')
|
2016-11-18 08:46:17 +03:00
|
|
|
c.clear()
|
2016-11-23 14:55:38 +03:00
|
|
|
pb('\033]\x07', ('set_title', ''), ('set_icon', ''))
|
|
|
|
self.ae(c.titlebuf, ''), self.ae(c.iconbuf, '')
|
|
|
|
pb('\033]ab\x07', ('set_title', 'ab'), ('set_icon', 'ab'))
|
|
|
|
self.ae(c.titlebuf, 'ab'), self.ae(c.iconbuf, 'ab')
|
2016-11-18 08:46:17 +03:00
|
|
|
c.clear()
|
2016-11-23 14:55:38 +03:00
|
|
|
pb('\033]2;;;;\x07', ('set_title', ';;;'))
|
|
|
|
self.ae(c.titlebuf, ';;;')
|
|
|
|
pb('\033]110\x07', ('set_dynamic_color', ''))
|
|
|
|
self.ae(c.colorbuf, '')
|
2016-11-18 09:01:48 +03:00
|
|
|
|
2016-11-23 17:52:52 +03:00
|
|
|
def test_dcs_codes(self):
|
|
|
|
s = self.create_screen()
|
|
|
|
pb = partial(self.parse_bytes_dump, s)
|
2016-11-24 05:44:29 +03:00
|
|
|
pb('a\033P+q436f\x9cbcde', 'a', ('screen_request_capabilities', '436f'), 'bcde')
|
2016-11-23 17:52:52 +03:00
|
|
|
self.ae(str(s.line(0)), 'abcde')
|