Skip to content

Commit 58aba5d

Browse files
authored
Improved routine for terminal dimensions. Remove TerminalSize function
* Improved routine for terminal dimensions. Works on windows too. * Improve clarity or variable names * Remove TerminalSize function .... no longer needed Co-authored-by: Daniel Harrison <[email protected]>
1 parent 162b102 commit 58aba5d

File tree

2 files changed

+15
-54
lines changed

2 files changed

+15
-54
lines changed

tests/terminal_test.py

+4-26
Original file line numberDiff line numberDiff line change
@@ -27,15 +27,11 @@ class TerminalTest(unittest.TestCase):
2727

2828
def setUp(self):
2929
super(TerminalTest, self).setUp()
30-
self.environ_orig = terminal.os.environ
31-
self.open_orig = terminal.os.open
32-
self.terminal_orig = terminal.TerminalSize
30+
self._get_terminal_size_orig = terminal.shutil.get_terminal_size
3331

3432
def tearDown(self):
3533
super(TerminalTest, self).tearDown()
36-
terminal.os.environ = self.environ_orig
37-
terminal.os.open = self.open_orig
38-
terminal.TerminalSize = self.terminal_orig
34+
terminal.shutil.get_terminal_size = self._get_terminal_size_orig
3935

4036
def testAnsiCmd(self):
4137
self.assertEqual('\033[0m', terminal._AnsiCmd(['reset']))
@@ -66,22 +62,8 @@ def testEncloseAnsi(self):
6662
ansi_enclosed = '\001\033[5;32;44m\002ansi\001\033[0m\002 length'
6763
self.assertEqual(ansi_enclosed, terminal.EncloseAnsiText(ansi_text))
6864

69-
def testTerminalSize(self):
70-
# pylint: disable=unused-argument
71-
def StubOpen(args, *kwargs):
72-
raise IOError
73-
terminal.open = StubOpen
74-
terminal.os.environ = {}
75-
# Raise exceptions on ioctl and environ and assign a default.
76-
self.assertEqual((24, 80), terminal.TerminalSize())
77-
terminal.os.environ = {'LINES': 'bogus', 'COLUMNS': 'bogus'}
78-
self.assertEqual((24, 80), terminal.TerminalSize())
79-
# Still raise exception on ioctl and use environ.
80-
terminal.os.environ = {'LINES': '10', 'COLUMNS': '20'}
81-
self.assertEqual((10, 20), terminal.TerminalSize())
82-
8365
def testLineWrap(self):
84-
terminal.TerminalSize = lambda: (5, 11)
66+
terminal.shutil.get_terminal_size = lambda: (11, 5)
8567
text = ''
8668
self.assertEqual(text, terminal.LineWrap(text))
8769
text = 'one line'
@@ -145,19 +127,15 @@ def setUp(self):
145127
sys.stdout = FakeTerminal()
146128
self.get_ch_orig = terminal.Pager._GetCh
147129
terminal.Pager._GetCh = lambda self: 'q'
148-
self.ts_orig = terminal.TerminalSize
149-
terminal.TerminalSize = lambda: (24, 80)
150130

151131
self.p = terminal.Pager()
152132

153133
def tearDown(self):
154134
super(PagerTest, self).tearDown()
155135
terminal.Pager._GetCh = self.get_ch_orig
156-
terminal.TerminalSize = self.ts_orig
157136
sys.stdout = sys.__stdout__
158137

159138
def testPager(self):
160-
self.assertEqual(terminal.TerminalSize()[0], self.p._cli_lines)
161139

162140
self.p.Clear()
163141
self.assertEqual('', self.p._text)
@@ -170,7 +148,7 @@ def testPage(self):
170148
txt += '%d a random line of text here\n' % i
171149
self.p._text = txt
172150
self.p.Page()
173-
self.assertEqual(terminal.TerminalSize()[0]+2, sys.stdout.CountLines())
151+
self.assertEqual(self.p._cli_lines+2, sys.stdout.CountLines())
174152

175153
sys.stdout.output = ''
176154
self.p = terminal.Pager()

textfsm/terminal.py

+11-28
Original file line numberDiff line numberDiff line change
@@ -18,9 +18,8 @@
1818
"""Simple terminal related routines."""
1919

2020
import getopt
21-
import os
2221
import re
23-
import struct
22+
import shutil
2423
import sys
2524
import time
2625

@@ -32,8 +31,6 @@
3231
except (ImportError, ModuleNotFoundError):
3332
pass
3433

35-
__version__ = '0.1.1'
36-
3734
# ANSI, ISO/IEC 6429 escape sequences, SGR (Select Graphic Rendition) subset.
3835
SGR = {
3936
'reset': 0,
@@ -168,21 +165,6 @@ def EncloseAnsiText(text):
168165
return sgr_re.sub(lambda x: ANSI_START + x.group(1) + ANSI_END, text)
169166

170167

171-
def TerminalSize():
172-
"""Returns terminal length and width as a tuple."""
173-
try:
174-
with open(os.ctermid()) as tty_instance:
175-
length_width = struct.unpack(
176-
'hh', fcntl.ioctl(tty_instance.fileno(), termios.TIOCGWINSZ, '1234')
177-
)
178-
except (IOError, OSError, NameError):
179-
try:
180-
length_width = (int(os.environ['LINES']), int(os.environ['COLUMNS']))
181-
except (ValueError, KeyError):
182-
length_width = (24, 80)
183-
return length_width
184-
185-
186168
def LineWrap(text, omit_sgr=False):
187169
"""Break line to fit screen width, factoring in ANSI/SGR escape sequences.
188170
@@ -194,7 +176,7 @@ def LineWrap(text, omit_sgr=False):
194176
Text with additional line wraps inserted for lines grater than the width.
195177
"""
196178

197-
def _SplitWithSgr(text_line):
179+
def _SplitWithSgr(text_line, width):
198180
"""Tokenise the line so that the sgr sequences can be omitted."""
199181
token_list = sgr_re.split(text_line)
200182
text_line_list = []
@@ -226,20 +208,20 @@ def _SplitWithSgr(text_line):
226208

227209
# We don't use textwrap library here as it insists on removing
228210
# trailing/leading whitespace (pre 2.6).
229-
(_, width) = TerminalSize()
211+
(term_width, _) = shutil.get_terminal_size()
230212
text = str(text)
231213
text_multiline = []
232214
for text_line in text.splitlines():
233215
# Is this a line that needs splitting?
234-
while (omit_sgr and (len(StripAnsiText(text_line)) > width)) or (
235-
len(text_line) > width
216+
while (omit_sgr and (len(StripAnsiText(text_line)) > term_width)) or (
217+
len(text_line) > term_width
236218
):
237219
# If there are no sgr escape characters then do a straight split.
238220
if not omit_sgr:
239-
text_multiline.append(text_line[:width])
240-
text_line = text_line[width:]
221+
text_multiline.append(text_line[:term_width])
222+
text_line = text_line[term_width:]
241223
else:
242-
(multiline_line, text_line) = _SplitWithSgr(text_line)
224+
(multiline_line, text_line) = _SplitWithSgr(text_line, term_width)
243225
text_multiline.append(multiline_line)
244226
if text_line:
245227
text_multiline.append(text_line)
@@ -318,7 +300,7 @@ def SetLines(self, lines):
318300
ValueError, TypeError: Not a valid integer representation.
319301
"""
320302

321-
(self._cli_lines, self._cli_cols) = TerminalSize()
303+
(self._cli_cols, self._cli_lines) = shutil.get_terminal_size()
322304

323305
if lines:
324306
self._cli_lines = int(lines)
@@ -472,7 +454,8 @@ def main(argv=None):
472454
# Prints the size of the terminal and returns.
473455
# Mutually exclusive to the paging of text and overrides that behaviour.
474456
if opt in ('-s', '--size'):
475-
print('Length: %d, Width: %d' % TerminalSize())
457+
print(
458+
'Width: %d, Length: %d' % shutil.get_terminal_size())
476459
return 0
477460
elif opt in ('-d', '--delay'):
478461
isdelay = True

0 commit comments

Comments
 (0)