123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342 |
- #!/usr/bin/env python
- # encoding: utf-8
- """
- Emulate a vt100 terminal in cmd.exe
- By wrapping sys.stdout / sys.stderr with Ansiterm,
- the vt100 escape characters will be interpreted and
- the equivalent actions will be performed with Win32
- console commands.
- """
- import os, re, sys
- from waflib import Utils
- wlock = Utils.threading.Lock()
- try:
- from ctypes import Structure, windll, c_short, c_ushort, c_ulong, c_int, byref, c_wchar, POINTER, c_long
- except ImportError:
- class AnsiTerm(object):
- def __init__(self, stream):
- self.stream = stream
- try:
- self.errors = self.stream.errors
- except AttributeError:
- pass # python 2.5
- self.encoding = self.stream.encoding
- def write(self, txt):
- try:
- wlock.acquire()
- self.stream.write(txt)
- self.stream.flush()
- finally:
- wlock.release()
- def fileno(self):
- return self.stream.fileno()
- def flush(self):
- self.stream.flush()
- def isatty(self):
- return self.stream.isatty()
- else:
- class COORD(Structure):
- _fields_ = [("X", c_short), ("Y", c_short)]
- class SMALL_RECT(Structure):
- _fields_ = [("Left", c_short), ("Top", c_short), ("Right", c_short), ("Bottom", c_short)]
- class CONSOLE_SCREEN_BUFFER_INFO(Structure):
- _fields_ = [("Size", COORD), ("CursorPosition", COORD), ("Attributes", c_ushort), ("Window", SMALL_RECT), ("MaximumWindowSize", COORD)]
- class CONSOLE_CURSOR_INFO(Structure):
- _fields_ = [('dwSize', c_ulong), ('bVisible', c_int)]
- try:
- _type = unicode
- except NameError:
- _type = str
- to_int = lambda number, default: number and int(number) or default
- STD_OUTPUT_HANDLE = -11
- STD_ERROR_HANDLE = -12
- windll.kernel32.GetStdHandle.argtypes = [c_ulong]
- windll.kernel32.GetStdHandle.restype = c_ulong
- windll.kernel32.GetConsoleScreenBufferInfo.argtypes = [c_ulong, POINTER(CONSOLE_SCREEN_BUFFER_INFO)]
- windll.kernel32.GetConsoleScreenBufferInfo.restype = c_long
- windll.kernel32.SetConsoleTextAttribute.argtypes = [c_ulong, c_ushort]
- windll.kernel32.SetConsoleTextAttribute.restype = c_long
- windll.kernel32.FillConsoleOutputCharacterW.argtypes = [c_ulong, c_wchar, c_ulong, POINTER(COORD), POINTER(c_ulong)]
- windll.kernel32.FillConsoleOutputCharacterW.restype = c_long
- windll.kernel32.FillConsoleOutputAttribute.argtypes = [c_ulong, c_ushort, c_ulong, POINTER(COORD), POINTER(c_ulong) ]
- windll.kernel32.FillConsoleOutputAttribute.restype = c_long
- windll.kernel32.SetConsoleCursorPosition.argtypes = [c_ulong, POINTER(COORD) ]
- windll.kernel32.SetConsoleCursorPosition.restype = c_long
- windll.kernel32.SetConsoleCursorInfo.argtypes = [c_ulong, POINTER(CONSOLE_CURSOR_INFO)]
- windll.kernel32.SetConsoleCursorInfo.restype = c_long
- class AnsiTerm(object):
- """
- emulate a vt100 terminal in cmd.exe
- """
- def __init__(self, s):
- self.stream = s
- try:
- self.errors = s.errors
- except AttributeError:
- pass # python2.5
- self.encoding = s.encoding
- self.cursor_history = []
- handle = (s.fileno() == 2) and STD_ERROR_HANDLE or STD_OUTPUT_HANDLE
- self.hconsole = windll.kernel32.GetStdHandle(handle)
- self._sbinfo = CONSOLE_SCREEN_BUFFER_INFO()
- self._csinfo = CONSOLE_CURSOR_INFO()
- windll.kernel32.GetConsoleCursorInfo(self.hconsole, byref(self._csinfo))
- # just to double check that the console is usable
- self._orig_sbinfo = CONSOLE_SCREEN_BUFFER_INFO()
- r = windll.kernel32.GetConsoleScreenBufferInfo(self.hconsole, byref(self._orig_sbinfo))
- self._isatty = r == 1
- def screen_buffer_info(self):
- """
- Updates self._sbinfo and returns it
- """
- windll.kernel32.GetConsoleScreenBufferInfo(self.hconsole, byref(self._sbinfo))
- return self._sbinfo
- def clear_line(self, param):
- mode = param and int(param) or 0
- sbinfo = self.screen_buffer_info()
- if mode == 1: # Clear from beginning of line to cursor position
- line_start = COORD(0, sbinfo.CursorPosition.Y)
- line_length = sbinfo.Size.X
- elif mode == 2: # Clear entire line
- line_start = COORD(sbinfo.CursorPosition.X, sbinfo.CursorPosition.Y)
- line_length = sbinfo.Size.X - sbinfo.CursorPosition.X
- else: # Clear from cursor position to end of line
- line_start = sbinfo.CursorPosition
- line_length = sbinfo.Size.X - sbinfo.CursorPosition.X
- chars_written = c_ulong()
- windll.kernel32.FillConsoleOutputCharacterW(self.hconsole, c_wchar(' '), line_length, line_start, byref(chars_written))
- windll.kernel32.FillConsoleOutputAttribute(self.hconsole, sbinfo.Attributes, line_length, line_start, byref(chars_written))
- def clear_screen(self, param):
- mode = to_int(param, 0)
- sbinfo = self.screen_buffer_info()
- if mode == 1: # Clear from beginning of screen to cursor position
- clear_start = COORD(0, 0)
- clear_length = sbinfo.CursorPosition.X * sbinfo.CursorPosition.Y
- elif mode == 2: # Clear entire screen and return cursor to home
- clear_start = COORD(0, 0)
- clear_length = sbinfo.Size.X * sbinfo.Size.Y
- windll.kernel32.SetConsoleCursorPosition(self.hconsole, clear_start)
- else: # Clear from cursor position to end of screen
- clear_start = sbinfo.CursorPosition
- clear_length = ((sbinfo.Size.X - sbinfo.CursorPosition.X) + sbinfo.Size.X * (sbinfo.Size.Y - sbinfo.CursorPosition.Y))
- chars_written = c_ulong()
- windll.kernel32.FillConsoleOutputCharacterW(self.hconsole, c_wchar(' '), clear_length, clear_start, byref(chars_written))
- windll.kernel32.FillConsoleOutputAttribute(self.hconsole, sbinfo.Attributes, clear_length, clear_start, byref(chars_written))
- def push_cursor(self, param):
- sbinfo = self.screen_buffer_info()
- self.cursor_history.append(sbinfo.CursorPosition)
- def pop_cursor(self, param):
- if self.cursor_history:
- old_pos = self.cursor_history.pop()
- windll.kernel32.SetConsoleCursorPosition(self.hconsole, old_pos)
- def set_cursor(self, param):
- y, sep, x = param.partition(';')
- x = to_int(x, 1) - 1
- y = to_int(y, 1) - 1
- sbinfo = self.screen_buffer_info()
- new_pos = COORD(
- min(max(0, x), sbinfo.Size.X),
- min(max(0, y), sbinfo.Size.Y)
- )
- windll.kernel32.SetConsoleCursorPosition(self.hconsole, new_pos)
- def set_column(self, param):
- x = to_int(param, 1) - 1
- sbinfo = self.screen_buffer_info()
- new_pos = COORD(
- min(max(0, x), sbinfo.Size.X),
- sbinfo.CursorPosition.Y
- )
- windll.kernel32.SetConsoleCursorPosition(self.hconsole, new_pos)
- def move_cursor(self, x_offset=0, y_offset=0):
- sbinfo = self.screen_buffer_info()
- new_pos = COORD(
- min(max(0, sbinfo.CursorPosition.X + x_offset), sbinfo.Size.X),
- min(max(0, sbinfo.CursorPosition.Y + y_offset), sbinfo.Size.Y)
- )
- windll.kernel32.SetConsoleCursorPosition(self.hconsole, new_pos)
- def move_up(self, param):
- self.move_cursor(y_offset = -to_int(param, 1))
- def move_down(self, param):
- self.move_cursor(y_offset = to_int(param, 1))
- def move_left(self, param):
- self.move_cursor(x_offset = -to_int(param, 1))
- def move_right(self, param):
- self.move_cursor(x_offset = to_int(param, 1))
- def next_line(self, param):
- sbinfo = self.screen_buffer_info()
- self.move_cursor(
- x_offset = -sbinfo.CursorPosition.X,
- y_offset = to_int(param, 1)
- )
- def prev_line(self, param):
- sbinfo = self.screen_buffer_info()
- self.move_cursor(
- x_offset = -sbinfo.CursorPosition.X,
- y_offset = -to_int(param, 1)
- )
- def rgb2bgr(self, c):
- return ((c&1) << 2) | (c&2) | ((c&4)>>2)
- def set_color(self, param):
- cols = param.split(';')
- sbinfo = self.screen_buffer_info()
- attr = sbinfo.Attributes
- for c in cols:
- c = to_int(c, 0)
- if 29 < c < 38: # fgcolor
- attr = (attr & 0xfff0) | self.rgb2bgr(c - 30)
- elif 39 < c < 48: # bgcolor
- attr = (attr & 0xff0f) | (self.rgb2bgr(c - 40) << 4)
- elif c == 0: # reset
- attr = self._orig_sbinfo.Attributes
- elif c == 1: # strong
- attr |= 0x08
- elif c == 4: # blink not available -> bg intensity
- attr |= 0x80
- elif c == 7: # negative
- attr = (attr & 0xff88) | ((attr & 0x70) >> 4) | ((attr & 0x07) << 4)
- windll.kernel32.SetConsoleTextAttribute(self.hconsole, attr)
- def show_cursor(self,param):
- self._csinfo.bVisible = 1
- windll.kernel32.SetConsoleCursorInfo(self.hconsole, byref(self._csinfo))
- def hide_cursor(self,param):
- self._csinfo.bVisible = 0
- windll.kernel32.SetConsoleCursorInfo(self.hconsole, byref(self._csinfo))
- ansi_command_table = {
- 'A': move_up,
- 'B': move_down,
- 'C': move_right,
- 'D': move_left,
- 'E': next_line,
- 'F': prev_line,
- 'G': set_column,
- 'H': set_cursor,
- 'f': set_cursor,
- 'J': clear_screen,
- 'K': clear_line,
- 'h': show_cursor,
- 'l': hide_cursor,
- 'm': set_color,
- 's': push_cursor,
- 'u': pop_cursor,
- }
- # Match either the escape sequence or text not containing escape sequence
- ansi_tokens = re.compile('(?:\x1b\[([0-9?;]*)([a-zA-Z])|([^\x1b]+))')
- def write(self, text):
- try:
- wlock.acquire()
- if self._isatty:
- for param, cmd, txt in self.ansi_tokens.findall(text):
- if cmd:
- cmd_func = self.ansi_command_table.get(cmd)
- if cmd_func:
- cmd_func(self, param)
- else:
- self.writeconsole(txt)
- else:
- # no support for colors in the console, just output the text:
- # eclipse or msys may be able to interpret the escape sequences
- self.stream.write(text)
- finally:
- wlock.release()
- def writeconsole(self, txt):
- chars_written = c_ulong()
- writeconsole = windll.kernel32.WriteConsoleA
- if isinstance(txt, _type):
- writeconsole = windll.kernel32.WriteConsoleW
- # MSDN says that there is a shared buffer of 64 KB for the console
- # writes. Attempt to not get ERROR_NOT_ENOUGH_MEMORY, see waf issue #746
- done = 0
- todo = len(txt)
- chunk = 32<<10
- while todo != 0:
- doing = min(chunk, todo)
- buf = txt[done:done+doing]
- r = writeconsole(self.hconsole, buf, doing, byref(chars_written), None)
- if r == 0:
- chunk >>= 1
- continue
- done += doing
- todo -= doing
- def fileno(self):
- return self.stream.fileno()
- def flush(self):
- pass
- def isatty(self):
- return self._isatty
- if sys.stdout.isatty() or sys.stderr.isatty():
- handle = sys.stdout.isatty() and STD_OUTPUT_HANDLE or STD_ERROR_HANDLE
- console = windll.kernel32.GetStdHandle(handle)
- sbinfo = CONSOLE_SCREEN_BUFFER_INFO()
- def get_term_cols():
- windll.kernel32.GetConsoleScreenBufferInfo(console, byref(sbinfo))
- # Issue 1401 - the progress bar cannot reach the last character
- return sbinfo.Size.X - 1
- # just try and see
- try:
- import struct, fcntl, termios
- except ImportError:
- pass
- else:
- if (sys.stdout.isatty() or sys.stderr.isatty()) and os.environ.get('TERM', '') not in ('dumb', 'emacs'):
- FD = sys.stdout.isatty() and sys.stdout.fileno() or sys.stderr.fileno()
- def fun():
- return struct.unpack("HHHH", fcntl.ioctl(FD, termios.TIOCGWINSZ, struct.pack("HHHH", 0, 0, 0, 0)))[1]
- try:
- fun()
- except Exception as e:
- pass
- else:
- get_term_cols = fun
|