Commit 0eb39e78 authored by Serhiy Storchaka's avatar Serhiy Storchaka

Issue #27067: Improved curses tests.

parent 732ba821
...@@ -24,19 +24,39 @@ requires('curses') ...@@ -24,19 +24,39 @@ requires('curses')
# If either of these don't exist, skip the tests. # If either of these don't exist, skip the tests.
curses = import_module('curses') curses = import_module('curses')
curses.panel = import_module('curses.panel') import_module('curses.panel')
import_module('curses.ascii')
term = os.environ.get('TERM', 'unknown') def requires_curses_func(name):
return unittest.skipUnless(hasattr(curses, name),
'requires curses.%s' % name)
@unittest.skipUnless(sys.__stdout__.isatty(), 'sys.__stdout__ is not a tty') term = os.environ.get('TERM')
@unittest.skipIf(term == 'unknown',
# If newterm was supported we could use it instead of initscr and not exit
@unittest.skipIf(not term or term == 'unknown',
"$TERM=%r, calling initscr() may cause exit" % term) "$TERM=%r, calling initscr() may cause exit" % term)
@unittest.skipIf(sys.platform == "cygwin", @unittest.skipIf(sys.platform == "cygwin",
"cygwin's curses mostly just hangs") "cygwin's curses mostly just hangs")
class TestCurses(unittest.TestCase): class TestCurses(unittest.TestCase):
@classmethod @classmethod
def setUpClass(cls): def setUpClass(cls):
curses.setupterm(fd=sys.__stdout__.fileno()) if not sys.__stdout__.isatty():
cls.tmp = tempfile.TemporaryFile()
fd = cls.tmp.fileno()
else:
cls.tmp = None
fd = sys.__stdout__.fileno()
# testing setupterm() inside initscr/endwin
# causes terminal breakage
curses.setupterm(fd=fd)
@classmethod
def tearDownClass(cls):
if cls.tmp:
cls.tmp.close()
del cls.tmp
def setUp(self): def setUp(self):
if verbose: if verbose:
...@@ -59,6 +79,7 @@ class TestCurses(unittest.TestCase): ...@@ -59,6 +79,7 @@ class TestCurses(unittest.TestCase):
for meth in [stdscr.addch, stdscr.addstr]: for meth in [stdscr.addch, stdscr.addstr]:
for args in [('a'), ('a', curses.A_BOLD), for args in [('a'), ('a', curses.A_BOLD),
(4,4, 'a'), (5,5, 'a', curses.A_BOLD)]: (4,4, 'a'), (5,5, 'a', curses.A_BOLD)]:
with self.subTest(meth=meth.__qualname__, args=args):
meth(*args) meth(*args)
for meth in [stdscr.box, stdscr.clear, stdscr.clrtobot, for meth in [stdscr.box, stdscr.clear, stdscr.clrtobot,
...@@ -70,6 +91,7 @@ class TestCurses(unittest.TestCase): ...@@ -70,6 +91,7 @@ class TestCurses(unittest.TestCase):
win.noutrefresh, stdscr.redrawwin, stdscr.refresh, win.noutrefresh, stdscr.redrawwin, stdscr.refresh,
stdscr.standout, stdscr.standend, stdscr.syncdown, stdscr.standout, stdscr.standend, stdscr.syncdown,
stdscr.syncup, stdscr.touchwin, stdscr.untouchwin]: stdscr.syncup, stdscr.touchwin, stdscr.untouchwin]:
with self.subTest(meth=meth.__qualname__):
meth() meth()
stdscr.addnstr('1234', 3) stdscr.addnstr('1234', 3)
...@@ -166,7 +188,6 @@ class TestCurses(unittest.TestCase): ...@@ -166,7 +188,6 @@ class TestCurses(unittest.TestCase):
def test_module_funcs(self): def test_module_funcs(self):
"Test module-level functions" "Test module-level functions"
stdscr = self.stdscr
for func in [curses.baudrate, curses.beep, curses.can_change_color, for func in [curses.baudrate, curses.beep, curses.can_change_color,
curses.cbreak, curses.def_prog_mode, curses.doupdate, curses.cbreak, curses.def_prog_mode, curses.doupdate,
curses.filter, curses.flash, curses.flushinp, curses.filter, curses.flash, curses.flushinp,
...@@ -176,6 +197,7 @@ class TestCurses(unittest.TestCase): ...@@ -176,6 +197,7 @@ class TestCurses(unittest.TestCase):
curses.noqiflush, curses.noraw, curses.noqiflush, curses.noraw,
curses.reset_prog_mode, curses.termattrs, curses.reset_prog_mode, curses.termattrs,
curses.termname, curses.erasechar, curses.getsyx]: curses.termname, curses.erasechar, curses.getsyx]:
with self.subTest(func=func.__qualname__):
func() func()
# Functions that actually need arguments # Functions that actually need arguments
...@@ -184,11 +206,10 @@ class TestCurses(unittest.TestCase): ...@@ -184,11 +206,10 @@ class TestCurses(unittest.TestCase):
curses.delay_output(1) curses.delay_output(1)
curses.echo() ; curses.echo(1) curses.echo() ; curses.echo(1)
f = tempfile.TemporaryFile() with tempfile.TemporaryFile() as f:
stdscr.putwin(f) self.stdscr.putwin(f)
f.seek(0) f.seek(0)
curses.getwin(f) curses.getwin(f)
f.close()
curses.halfdelay(1) curses.halfdelay(1)
curses.intrflush(1) curses.intrflush(1)
...@@ -212,7 +233,9 @@ class TestCurses(unittest.TestCase): ...@@ -212,7 +233,9 @@ class TestCurses(unittest.TestCase):
curses.use_env(1) curses.use_env(1)
# Functions only available on a few platforms # Functions only available on a few platforms
if curses.has_colors(): def test_colors_funcs(self):
if not curses.has_colors():
self.skip('requires colors support')
curses.start_color() curses.start_color()
curses.init_pair(2, 1,1) curses.init_pair(2, 1,1)
curses.color_content(1) curses.color_content(1)
...@@ -223,40 +246,24 @@ class TestCurses(unittest.TestCase): ...@@ -223,40 +246,24 @@ class TestCurses(unittest.TestCase):
if hasattr(curses, 'use_default_colors'): if hasattr(curses, 'use_default_colors'):
curses.use_default_colors() curses.use_default_colors()
if hasattr(curses, 'keyname'): @requires_curses_func('keyname')
def test_keyname(self):
curses.keyname(13) curses.keyname(13)
if hasattr(curses, 'has_key'): @requires_curses_func('has_key')
def test_has_key(self):
curses.has_key(13) curses.has_key(13)
if hasattr(curses, 'getmouse'): @requires_curses_func('getmouse')
def test_getmouse(self):
(availmask, oldmask) = curses.mousemask(curses.BUTTON1_PRESSED) (availmask, oldmask) = curses.mousemask(curses.BUTTON1_PRESSED)
# availmask indicates that mouse stuff not available. if availmask == 0:
if availmask != 0: self.skip('mouse stuff not available')
curses.mouseinterval(10) curses.mouseinterval(10)
# just verify these don't cause errors # just verify these don't cause errors
curses.ungetmouse(0, 0, 0, 0, curses.BUTTON1_PRESSED) curses.ungetmouse(0, 0, 0, 0, curses.BUTTON1_PRESSED)
m = curses.getmouse() m = curses.getmouse()
if hasattr(curses, 'is_term_resized'):
curses.is_term_resized(*stdscr.getmaxyx())
if hasattr(curses, 'resizeterm'):
curses.resizeterm(*stdscr.getmaxyx())
if hasattr(curses, 'resize_term'):
curses.resize_term(*stdscr.getmaxyx())
def test_unctrl(self):
from curses import ascii
for ch, expected in [('a', 'a'), ('A', 'A'),
(';', ';'), (' ', ' '),
('\x7f', '^?'), ('\n', '^J'), ('\0', '^@'),
# Meta-bit characters
('\x8a', '!^J'), ('\xc1', '!A'),
]:
self.assertEqual(ascii.unctrl(ch), expected,
'curses.unctrl fails on character %r' % ch)
def test_userptr_without_set(self): def test_userptr_without_set(self):
w = curses.newwin(10, 10) w = curses.newwin(10, 10)
p = curses.panel.new_panel(w) p = curses.panel.new_panel(w)
...@@ -289,9 +296,17 @@ class TestCurses(unittest.TestCase): ...@@ -289,9 +296,17 @@ class TestCurses(unittest.TestCase):
panel = curses.panel.new_panel(self.stdscr) panel = curses.panel.new_panel(self.stdscr)
self.assertRaises(TypeError, type(panel)) self.assertRaises(TypeError, type(panel))
@unittest.skipUnless(hasattr(curses, 'resizeterm'), @requires_curses_func('is_term_resized')
'resizeterm not available') def test_is_term_resized(self):
curses.is_term_resized(*self.stdscr.getmaxyx())
@requires_curses_func('resize_term')
def test_resize_term(self): def test_resize_term(self):
curses.resize_term(*self.stdscr.getmaxyx())
@requires_curses_func('resizeterm')
def test_resizeterm(self):
stdscr = self.stdscr
lines, cols = curses.LINES, curses.COLS lines, cols = curses.LINES, curses.COLS
new_lines = lines - 1 new_lines = lines - 1
new_cols = cols + 1 new_cols = cols + 1
...@@ -304,8 +319,7 @@ class TestCurses(unittest.TestCase): ...@@ -304,8 +319,7 @@ class TestCurses(unittest.TestCase):
curses.ungetch(1025) curses.ungetch(1025)
self.stdscr.getkey() self.stdscr.getkey()
@unittest.skipUnless(hasattr(curses, 'unget_wch'), @requires_curses_func('unget_wch')
'unget_wch not available')
def test_unget_wch(self): def test_unget_wch(self):
stdscr = self.stdscr stdscr = self.stdscr
encoding = stdscr.encoding encoding = stdscr.encoding
...@@ -330,17 +344,14 @@ class TestCurses(unittest.TestCase): ...@@ -330,17 +344,14 @@ class TestCurses(unittest.TestCase):
def test_issue10570(self): def test_issue10570(self):
b = curses.tparm(curses.tigetstr("cup"), 5, 3) b = curses.tparm(curses.tigetstr("cup"), 5, 3)
self.assertIs(type(b), bytes) self.assertIs(type(b), bytes)
curses.putp(b)
def test_encoding(self): def test_encoding(self):
stdscr = self.stdscr stdscr = self.stdscr
import codecs import codecs
encoding = stdscr.encoding encoding = stdscr.encoding
codecs.lookup(encoding) codecs.lookup(encoding)
with self.assertRaises(TypeError): with self.assertRaises(TypeError):
stdscr.encoding = 10 stdscr.encoding = 10
stdscr.encoding = encoding stdscr.encoding = encoding
with self.assertRaises(TypeError): with self.assertRaises(TypeError):
del stdscr.encoding del stdscr.encoding
...@@ -371,8 +382,10 @@ class TestCurses(unittest.TestCase): ...@@ -371,8 +382,10 @@ class TestCurses(unittest.TestCase):
# be reasonably certain the generated parsing code will be # be reasonably certain the generated parsing code will be
# correct too. # correct too.
human_readable_signature = stdscr.addch.__doc__.split("\n")[0] human_readable_signature = stdscr.addch.__doc__.split("\n")[0]
offset = human_readable_signature.find("[y, x,]") self.assertIn("[y, x,]", human_readable_signature)
assert offset >= 0, ""
class MiscTests(unittest.TestCase):
def test_update_lines_cols(self): def test_update_lines_cols(self):
# this doesn't actually test that LINES and COLS are updated, # this doesn't actually test that LINES and COLS are updated,
...@@ -382,5 +395,21 @@ class TestCurses(unittest.TestCase): ...@@ -382,5 +395,21 @@ class TestCurses(unittest.TestCase):
curses.update_lines_cols() curses.update_lines_cols()
class TestAscii(unittest.TestCase):
def test_unctrl(self):
unctrl = curses.ascii.unctrl
self.assertEqual(unctrl('a'), 'a')
self.assertEqual(unctrl('A'), 'A')
self.assertEqual(unctrl(';'), ';')
self.assertEqual(unctrl(' '), ' ')
self.assertEqual(unctrl('\x7f'), '^?')
self.assertEqual(unctrl('\n'), '^J')
self.assertEqual(unctrl('\0'), '^@')
# Meta-bit characters
self.assertEqual(unctrl('\x8a'), '!^J')
self.assertEqual(unctrl('\xc1'), '!A')
if __name__ == '__main__': if __name__ == '__main__':
unittest.main() unittest.main()
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment