mirror of
https://github.com/clearml/wexpect-venv
synced 2025-01-31 10:57:07 +00:00
192 lines
6.4 KiB
Python
192 lines
6.4 KiB
Python
"""Wexpect is a Windows variant of pexpect https://pexpect.readthedocs.io.
|
|
|
|
Wexpect is a Python module for spawning child applications and controlling
|
|
them automatically.
|
|
|
|
wexpect util contains small functions, and classes, which are used in multiple classes.
|
|
The command line argument parsers, and the Exceptions placed here.
|
|
|
|
Credits: Noah Spurrier, Richard Holden, Marco Molteni, Kimberley Burchett,
|
|
Robert Stone, Hartmut Goebel, Chad Schroeder, Erick Tryzelaar, Dave Kirby, Ids
|
|
vander Molen, George Todd, Noel Taylor, Nicolas D. Cesar, Alexander Gattin,
|
|
Geoffrey Marshall, Francisco Lourenco, Glen Mabey, Karthik Gurusamy, Fernando
|
|
Perez, Corey Minyard, Jon Cohen, Guillaume Chazarain, Andrew Ryan, Nick
|
|
Craig-Wood, Andrew Stone, Jorgen Grahn, Benedek Racz
|
|
|
|
Free, open source, and all that good stuff.
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy of
|
|
this software and associated documentation files (the "Software"), to deal in
|
|
the Software without restriction, including without limitation the rights to
|
|
use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
|
|
of the Software, and to permit persons to whom the Software is furnished to do
|
|
so, subject to the following conditions:
|
|
|
|
The above copyright notice and this permission notice shall be included in all
|
|
copies or substantial portions of the Software.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
SOFTWARE.
|
|
|
|
Wexpect Copyright (c) 2019 Benedek Racz
|
|
|
|
"""
|
|
|
|
import re
|
|
import traceback
|
|
import sys
|
|
import os
|
|
import logging
|
|
import signal
|
|
|
|
# platform does not define VEOF so assume CTRL-D
|
|
EOF_CHAR = b'\x04'
|
|
|
|
SIGNAL_CHARS = {
|
|
signal.SIGTERM: b'\x011', # Device control 1
|
|
signal.SIGINT: b'\x012', # Device control 2
|
|
}
|
|
|
|
SPAM = 5
|
|
logging.addLevelName(SPAM, "SPAM")
|
|
|
|
|
|
def spam(self, message, *args, **kws):
|
|
if self.isEnabledFor(SPAM):
|
|
# Yes, logger takes its '*args' as 'args'.
|
|
self._log(SPAM, message, args, **kws)
|
|
|
|
|
|
logging.Logger.spam = spam
|
|
|
|
|
|
def init_logger(logger):
|
|
try:
|
|
logger_level = os.environ['WEXPECT_LOGGER_LEVEL']
|
|
try:
|
|
logger_filename = os.environ['WEXPECT_LOGGER_FILENAME']
|
|
except KeyError:
|
|
pid = os.getpid()
|
|
logger_filename = f'./.wlog/wexpect_{pid}'
|
|
logger.setLevel(logger_level)
|
|
logger_filename = f'{logger_filename}.log'
|
|
os.makedirs(os.path.dirname(logger_filename), exist_ok=True)
|
|
fh = logging.FileHandler(logger_filename, 'w', 'utf-8')
|
|
formatter = logging.Formatter(
|
|
'%(asctime)s - %(filename)s:%(lineno)d - %(levelname)s - %(message)s')
|
|
fh.setFormatter(formatter)
|
|
logger.addHandler(fh)
|
|
except KeyError:
|
|
logger.setLevel(logging.ERROR)
|
|
|
|
|
|
def split_command_line(command_line, escape_char='^'):
|
|
"""This splits a command line into a list of arguments. It splits arguments
|
|
on spaces, but handles embedded quotes, doublequotes, and escaped
|
|
characters. It's impossible to do this with a regular expression, so I
|
|
wrote a little state machine to parse the command line. """
|
|
|
|
arg_list = []
|
|
arg = ''
|
|
|
|
# Constants to name the states we can be in.
|
|
state_basic = 0
|
|
state_esc = 1
|
|
state_singlequote = 2
|
|
state_doublequote = 3
|
|
state_whitespace = 4 # The state of consuming whitespace between commands.
|
|
state = state_basic
|
|
|
|
for c in command_line:
|
|
if state == state_basic or state == state_whitespace:
|
|
if c == escape_char: # Escape the next character
|
|
state = state_esc
|
|
elif c == r"'": # Handle single quote
|
|
state = state_singlequote
|
|
elif c == r'"': # Handle double quote
|
|
state = state_doublequote
|
|
elif c.isspace():
|
|
# Add arg to arg_list if we aren't in the middle of whitespace.
|
|
if state == state_whitespace:
|
|
None # Do nothing.
|
|
else:
|
|
arg_list.append(arg)
|
|
arg = ''
|
|
state = state_whitespace
|
|
else:
|
|
arg = arg + c
|
|
state = state_basic
|
|
elif state == state_esc:
|
|
arg = arg + c
|
|
state = state_basic
|
|
elif state == state_singlequote:
|
|
if c == r"'":
|
|
state = state_basic
|
|
else:
|
|
arg = arg + c
|
|
elif state == state_doublequote:
|
|
if c == r'"':
|
|
state = state_basic
|
|
else:
|
|
arg = arg + c
|
|
|
|
if arg != '':
|
|
arg_list.append(arg)
|
|
return arg_list
|
|
|
|
|
|
def join_args(args):
|
|
"""Joins arguments a command line. It quotes all arguments that contain
|
|
spaces or any of the characters ^!$%&()[]{}=;'+,`~"""
|
|
commandline = []
|
|
for arg in args:
|
|
if re.search('[\\^!$%&()[\\]{}=;\'+,`~\\s]', arg):
|
|
arg = '"%s"' % arg
|
|
commandline.append(arg)
|
|
return ' '.join(commandline)
|
|
|
|
|
|
class ExceptionPexpect(Exception):
|
|
"""Base class for all exceptions raised by this module.
|
|
"""
|
|
|
|
def __init__(self, value):
|
|
|
|
self.value = value
|
|
|
|
def __str__(self):
|
|
|
|
return str(self.value)
|
|
|
|
def get_trace(self):
|
|
"""This returns an abbreviated stack trace with lines that only concern
|
|
the caller. In other words, the stack trace inside the Wexpect module
|
|
is not included. """
|
|
|
|
tblist = traceback.extract_tb(sys.exc_info()[2])
|
|
tblist = [item for item in tblist if self.__filter_not_wexpect(item)]
|
|
tblist = traceback.format_list(tblist)
|
|
return ''.join(tblist)
|
|
|
|
def __filter_not_wexpect(self, trace_list_item):
|
|
"""This returns True if list item 0 the string 'wexpect.py' in it. """
|
|
|
|
if trace_list_item[0].find('host.py') == -1:
|
|
return True
|
|
else:
|
|
return False
|
|
|
|
|
|
class EOF(ExceptionPexpect):
|
|
"""Raised when EOF is read from a child. This usually means the child has exited.
|
|
The user can wait to EOF, which means he waits the end of the execution of the child process."""
|
|
|
|
|
|
class TIMEOUT(ExceptionPexpect):
|
|
"""Raised when a read time exceeds the timeout. """
|