Buckets:
MisterAI/LocalAI_Demo_backends / cpu-diffusers.upgrade-tmp /venv /lib /python3.10 /site-packages /pexpect /pty_spawn.py
| import os | |
| import sys | |
| import time | |
| import pty | |
| import tty | |
| import errno | |
| import signal | |
| from contextlib import contextmanager | |
| import ptyprocess | |
| from ptyprocess.ptyprocess import use_native_pty_fork | |
| from .exceptions import ExceptionPexpect, EOF, TIMEOUT | |
| from .spawnbase import SpawnBase | |
| from .utils import ( | |
| which, split_command_line, select_ignore_interrupts, poll_ignore_interrupts | |
| ) | |
| def _wrap_ptyprocess_err(): | |
| """Turn ptyprocess errors into our own ExceptionPexpect errors""" | |
| try: | |
| yield | |
| except ptyprocess.PtyProcessError as e: | |
| raise ExceptionPexpect(*e.args) | |
| PY3 = (sys.version_info[0] >= 3) | |
| class spawn(SpawnBase): | |
| '''This is the main class interface for Pexpect. Use this class to start | |
| and control child applications. ''' | |
| # This is purely informational now - changing it has no effect | |
| use_native_pty_fork = use_native_pty_fork | |
| def __init__(self, command, args=[], timeout=30, maxread=2000, | |
| searchwindowsize=None, logfile=None, cwd=None, env=None, | |
| ignore_sighup=False, echo=True, preexec_fn=None, | |
| encoding=None, codec_errors='strict', dimensions=None, | |
| use_poll=False): | |
| '''This is the constructor. The command parameter may be a string that | |
| includes a command and any arguments to the command. For example:: | |
| child = pexpect.spawn('/usr/bin/ftp') | |
| child = pexpect.spawn('/usr/bin/ssh user@example.com') | |
| child = pexpect.spawn('ls -latr /tmp') | |
| You may also construct it with a list of arguments like so:: | |
| child = pexpect.spawn('/usr/bin/ftp', []) | |
| child = pexpect.spawn('/usr/bin/ssh', ['user@example.com']) | |
| child = pexpect.spawn('ls', ['-latr', '/tmp']) | |
| After this the child application will be created and will be ready to | |
| talk to. For normal use, see expect() and send() and sendline(). | |
| Remember that Pexpect does NOT interpret shell meta characters such as | |
| redirect, pipe, or wild cards (``>``, ``|``, or ``*``). This is a | |
| common mistake. If you want to run a command and pipe it through | |
| another command then you must also start a shell. For example:: | |
| child = pexpect.spawn('/bin/bash -c "ls -l | grep LOG > logs.txt"') | |
| child.expect(pexpect.EOF) | |
| The second form of spawn (where you pass a list of arguments) is useful | |
| in situations where you wish to spawn a command and pass it its own | |
| argument list. This can make syntax more clear. For example, the | |
| following is equivalent to the previous example:: | |
| shell_cmd = 'ls -l | grep LOG > logs.txt' | |
| child = pexpect.spawn('/bin/bash', ['-c', shell_cmd]) | |
| child.expect(pexpect.EOF) | |
| The maxread attribute sets the read buffer size. This is maximum number | |
| of bytes that Pexpect will try to read from a TTY at one time. Setting | |
| the maxread size to 1 will turn off buffering. Setting the maxread | |
| value higher may help performance in cases where large amounts of | |
| output are read back from the child. This feature is useful in | |
| conjunction with searchwindowsize. | |
| When the keyword argument *searchwindowsize* is None (default), the | |
| full buffer is searched at each iteration of receiving incoming data. | |
| The default number of bytes scanned at each iteration is very large | |
| and may be reduced to collaterally reduce search cost. After | |
| :meth:`~.expect` returns, the full buffer attribute remains up to | |
| size *maxread* irrespective of *searchwindowsize* value. | |
| When the keyword argument ``timeout`` is specified as a number, | |
| (default: *30*), then :class:`TIMEOUT` will be raised after the value | |
| specified has elapsed, in seconds, for any of the :meth:`~.expect` | |
| family of method calls. When None, TIMEOUT will not be raised, and | |
| :meth:`~.expect` may block indefinitely until match. | |
| The logfile member turns on or off logging. All input and output will | |
| be copied to the given file object. Set logfile to None to stop | |
| logging. This is the default. Set logfile to sys.stdout to echo | |
| everything to standard output. The logfile is flushed after each write. | |
| Example log input and output to a file:: | |
| child = pexpect.spawn('some_command') | |
| fout = open('mylog.txt','wb') | |
| child.logfile = fout | |
| Example log to stdout:: | |
| # In Python 2: | |
| child = pexpect.spawn('some_command') | |
| child.logfile = sys.stdout | |
| # In Python 3, we'll use the ``encoding`` argument to decode data | |
| # from the subprocess and handle it as unicode: | |
| child = pexpect.spawn('some_command', encoding='utf-8') | |
| child.logfile = sys.stdout | |
| The logfile_read and logfile_send members can be used to separately log | |
| the input from the child and output sent to the child. Sometimes you | |
| don't want to see everything you write to the child. You only want to | |
| log what the child sends back. For example:: | |
| child = pexpect.spawn('some_command') | |
| child.logfile_read = sys.stdout | |
| You will need to pass an encoding to spawn in the above code if you are | |
| using Python 3. | |
| To separately log output sent to the child use logfile_send:: | |
| child.logfile_send = fout | |
| If ``ignore_sighup`` is True, the child process will ignore SIGHUP | |
| signals. The default is False from Pexpect 4.0, meaning that SIGHUP | |
| will be handled normally by the child. | |
| The delaybeforesend helps overcome a weird behavior that many users | |
| were experiencing. The typical problem was that a user would expect() a | |
| "Password:" prompt and then immediately call sendline() to send the | |
| password. The user would then see that their password was echoed back | |
| to them. Passwords don't normally echo. The problem is caused by the | |
| fact that most applications print out the "Password" prompt and then | |
| turn off stdin echo, but if you send your password before the | |
| application turned off echo, then you get your password echoed. | |
| Normally this wouldn't be a problem when interacting with a human at a | |
| real keyboard. If you introduce a slight delay just before writing then | |
| this seems to clear up the problem. This was such a common problem for | |
| many users that I decided that the default pexpect behavior should be | |
| to sleep just before writing to the child application. 1/20th of a | |
| second (50 ms) seems to be enough to clear up the problem. You can set | |
| delaybeforesend to None to return to the old behavior. | |
| Note that spawn is clever about finding commands on your path. | |
| It uses the same logic that "which" uses to find executables. | |
| If you wish to get the exit status of the child you must call the | |
| close() method. The exit or signal status of the child will be stored | |
| in self.exitstatus or self.signalstatus. If the child exited normally | |
| then exitstatus will store the exit return code and signalstatus will | |
| be None. If the child was terminated abnormally with a signal then | |
| signalstatus will store the signal value and exitstatus will be None:: | |
| child = pexpect.spawn('some_command') | |
| child.close() | |
| print(child.exitstatus, child.signalstatus) | |
| If you need more detail you can also read the self.status member which | |
| stores the status returned by os.waitpid. You can interpret this using | |
| os.WIFEXITED/os.WEXITSTATUS or os.WIFSIGNALED/os.TERMSIG. | |
| The echo attribute may be set to False to disable echoing of input. | |
| As a pseudo-terminal, all input echoed by the "keyboard" (send() | |
| or sendline()) will be repeated to output. For many cases, it is | |
| not desirable to have echo enabled, and it may be later disabled | |
| using setecho(False) followed by waitnoecho(). However, for some | |
| platforms such as Solaris, this is not possible, and should be | |
| disabled immediately on spawn. | |
| If preexec_fn is given, it will be called in the child process before | |
| launching the given command. This is useful to e.g. reset inherited | |
| signal handlers. | |
| The dimensions attribute specifies the size of the pseudo-terminal as | |
| seen by the subprocess, and is specified as a two-entry tuple (rows, | |
| columns). If this is unspecified, the defaults in ptyprocess will apply. | |
| The use_poll attribute enables using select.poll() over select.select() | |
| for socket handling. This is handy if your system could have > 1024 fds | |
| ''' | |
| super(spawn, self).__init__(timeout=timeout, maxread=maxread, searchwindowsize=searchwindowsize, | |
| logfile=logfile, encoding=encoding, codec_errors=codec_errors) | |
| self.STDIN_FILENO = pty.STDIN_FILENO | |
| self.STDOUT_FILENO = pty.STDOUT_FILENO | |
| self.STDERR_FILENO = pty.STDERR_FILENO | |
| self.str_last_chars = 100 | |
| self.cwd = cwd | |
| self.env = env | |
| self.echo = echo | |
| self.ignore_sighup = ignore_sighup | |
| self.__irix_hack = sys.platform.lower().startswith('irix') | |
| if command is None: | |
| self.command = None | |
| self.args = None | |
| self.name = '<pexpect factory incomplete>' | |
| else: | |
| self._spawn(command, args, preexec_fn, dimensions) | |
| self.use_poll = use_poll | |
| def __str__(self): | |
| '''This returns a human-readable string that represents the state of | |
| the object. ''' | |
| s = [] | |
| s.append(repr(self)) | |
| s.append('command: ' + str(self.command)) | |
| s.append('args: %r' % (self.args,)) | |
| s.append('buffer (last %s chars): %r' % (self.str_last_chars,self.buffer[-self.str_last_chars:])) | |
| s.append('before (last %s chars): %r' % (self.str_last_chars,self.before[-self.str_last_chars:] if self.before else '')) | |
| s.append('after: %r' % (self.after,)) | |
| s.append('match: %r' % (self.match,)) | |
| s.append('match_index: ' + str(self.match_index)) | |
| s.append('exitstatus: ' + str(self.exitstatus)) | |
| if hasattr(self, 'ptyproc'): | |
| s.append('flag_eof: ' + str(self.flag_eof)) | |
| s.append('pid: ' + str(self.pid)) | |
| s.append('child_fd: ' + str(self.child_fd)) | |
| s.append('closed: ' + str(self.closed)) | |
| s.append('timeout: ' + str(self.timeout)) | |
| s.append('delimiter: ' + str(self.delimiter)) | |
| s.append('logfile: ' + str(self.logfile)) | |
| s.append('logfile_read: ' + str(self.logfile_read)) | |
| s.append('logfile_send: ' + str(self.logfile_send)) | |
| s.append('maxread: ' + str(self.maxread)) | |
| s.append('ignorecase: ' + str(self.ignorecase)) | |
| s.append('searchwindowsize: ' + str(self.searchwindowsize)) | |
| s.append('delaybeforesend: ' + str(self.delaybeforesend)) | |
| s.append('delayafterclose: ' + str(self.delayafterclose)) | |
| s.append('delayafterterminate: ' + str(self.delayafterterminate)) | |
| return '\n'.join(s) | |
| def _spawn(self, command, args=[], preexec_fn=None, dimensions=None): | |
| '''This starts the given command in a child process. This does all the | |
| fork/exec type of stuff for a pty. This is called by __init__. If args | |
| is empty then command will be parsed (split on spaces) and args will be | |
| set to parsed arguments. ''' | |
| # The pid and child_fd of this object get set by this method. | |
| # Note that it is difficult for this method to fail. | |
| # You cannot detect if the child process cannot start. | |
| # So the only way you can tell if the child process started | |
| # or not is to try to read from the file descriptor. If you get | |
| # EOF immediately then it means that the child is already dead. | |
| # That may not necessarily be bad because you may have spawned a child | |
| # that performs some task; creates no stdout output; and then dies. | |
| # If command is an int type then it may represent a file descriptor. | |
| if isinstance(command, type(0)): | |
| raise ExceptionPexpect('Command is an int type. ' + | |
| 'If this is a file descriptor then maybe you want to ' + | |
| 'use fdpexpect.fdspawn which takes an existing ' + | |
| 'file descriptor instead of a command string.') | |
| if not isinstance(args, type([])): | |
| raise TypeError('The argument, args, must be a list.') | |
| if args == []: | |
| self.args = split_command_line(command) | |
| self.command = self.args[0] | |
| else: | |
| # Make a shallow copy of the args list. | |
| self.args = args[:] | |
| self.args.insert(0, command) | |
| self.command = command | |
| command_with_path = which(self.command, env=self.env) | |
| if command_with_path is None: | |
| raise ExceptionPexpect('The command was not found or was not ' + | |
| 'executable: %s.' % self.command) | |
| self.command = command_with_path | |
| self.args[0] = self.command | |
| self.name = '<' + ' '.join(self.args) + '>' | |
| assert self.pid is None, 'The pid member must be None.' | |
| assert self.command is not None, 'The command member must not be None.' | |
| kwargs = {'echo': self.echo, 'preexec_fn': preexec_fn} | |
| if self.ignore_sighup: | |
| def preexec_wrapper(): | |
| "Set SIGHUP to be ignored, then call the real preexec_fn" | |
| signal.signal(signal.SIGHUP, signal.SIG_IGN) | |
| if preexec_fn is not None: | |
| preexec_fn() | |
| kwargs['preexec_fn'] = preexec_wrapper | |
| if dimensions is not None: | |
| kwargs['dimensions'] = dimensions | |
| if self.encoding is not None: | |
| # Encode command line using the specified encoding | |
| self.args = [a if isinstance(a, bytes) else a.encode(self.encoding) | |
| for a in self.args] | |
| self.ptyproc = self._spawnpty(self.args, env=self.env, | |
| cwd=self.cwd, **kwargs) | |
| self.pid = self.ptyproc.pid | |
| self.child_fd = self.ptyproc.fd | |
| self.terminated = False | |
| self.closed = False | |
| def _spawnpty(self, args, **kwargs): | |
| '''Spawn a pty and return an instance of PtyProcess.''' | |
| return ptyprocess.PtyProcess.spawn(args, **kwargs) | |
| def close(self, force=True): | |
| '''This closes the connection with the child application. Note that | |
| calling close() more than once is valid. This emulates standard Python | |
| behavior with files. Set force to True if you want to make sure that | |
| the child is terminated (SIGKILL is sent if the child ignores SIGHUP | |
| and SIGINT). ''' | |
| self.flush() | |
| with _wrap_ptyprocess_err(): | |
| # PtyProcessError may be raised if it is not possible to terminate | |
| # the child. | |
| self.ptyproc.close(force=force) | |
| self.isalive() # Update exit status from ptyproc | |
| self.child_fd = -1 | |
| self.closed = True | |
| def isatty(self): | |
| '''This returns True if the file descriptor is open and connected to a | |
| tty(-like) device, else False. | |
| On SVR4-style platforms implementing streams, such as SunOS and HP-UX, | |
| the child pty may not appear as a terminal device. This means | |
| methods such as setecho(), setwinsize(), getwinsize() may raise an | |
| IOError. ''' | |
| return os.isatty(self.child_fd) | |
| def waitnoecho(self, timeout=-1): | |
| '''This waits until the terminal ECHO flag is set False. This returns | |
| True if the echo mode is off. This returns False if the ECHO flag was | |
| not set False before the timeout. This can be used to detect when the | |
| child is waiting for a password. Usually a child application will turn | |
| off echo mode when it is waiting for the user to enter a password. For | |
| example, instead of expecting the "password:" prompt you can wait for | |
| the child to set ECHO off:: | |
| p = pexpect.spawn('ssh user@example.com') | |
| p.waitnoecho() | |
| p.sendline(mypassword) | |
| If timeout==-1 then this method will use the value in self.timeout. | |
| If timeout==None then this method to block until ECHO flag is False. | |
| ''' | |
| if timeout == -1: | |
| timeout = self.timeout | |
| if timeout is not None: | |
| end_time = time.time() + timeout | |
| while True: | |
| if not self.getecho(): | |
| return True | |
| if timeout < 0 and timeout is not None: | |
| return False | |
| if timeout is not None: | |
| timeout = end_time - time.time() | |
| time.sleep(0.1) | |
| def getecho(self): | |
| '''This returns the terminal echo mode. This returns True if echo is | |
| on or False if echo is off. Child applications that are expecting you | |
| to enter a password often set ECHO False. See waitnoecho(). | |
| Not supported on platforms where ``isatty()`` returns False. ''' | |
| return self.ptyproc.getecho() | |
| def setecho(self, state): | |
| '''This sets the terminal echo mode on or off. Note that anything the | |
| child sent before the echo will be lost, so you should be sure that | |
| your input buffer is empty before you call setecho(). For example, the | |
| following will work as expected:: | |
| p = pexpect.spawn('cat') # Echo is on by default. | |
| p.sendline('1234') # We expect see this twice from the child... | |
| p.expect(['1234']) # ... once from the tty echo... | |
| p.expect(['1234']) # ... and again from cat itself. | |
| p.setecho(False) # Turn off tty echo | |
| p.sendline('abcd') # We will set this only once (echoed by cat). | |
| p.sendline('wxyz') # We will set this only once (echoed by cat) | |
| p.expect(['abcd']) | |
| p.expect(['wxyz']) | |
| The following WILL NOT WORK because the lines sent before the setecho | |
| will be lost:: | |
| p = pexpect.spawn('cat') | |
| p.sendline('1234') | |
| p.setecho(False) # Turn off tty echo | |
| p.sendline('abcd') # We will set this only once (echoed by cat). | |
| p.sendline('wxyz') # We will set this only once (echoed by cat) | |
| p.expect(['1234']) | |
| p.expect(['1234']) | |
| p.expect(['abcd']) | |
| p.expect(['wxyz']) | |
| Not supported on platforms where ``isatty()`` returns False. | |
| ''' | |
| return self.ptyproc.setecho(state) | |
| def read_nonblocking(self, size=1, timeout=-1): | |
| '''This reads at most size characters from the child application. It | |
| includes a timeout. If the read does not complete within the timeout | |
| period then a TIMEOUT exception is raised. If the end of file is read | |
| then an EOF exception will be raised. If a logfile is specified, a | |
| copy is written to that log. | |
| If timeout is None then the read may block indefinitely. | |
| If timeout is -1 then the self.timeout value is used. If timeout is 0 | |
| then the child is polled and if there is no data immediately ready | |
| then this will raise a TIMEOUT exception. | |
| The timeout refers only to the amount of time to read at least one | |
| character. This is not affected by the 'size' parameter, so if you call | |
| read_nonblocking(size=100, timeout=30) and only one character is | |
| available right away then one character will be returned immediately. | |
| It will not wait for 30 seconds for another 99 characters to come in. | |
| On the other hand, if there are bytes available to read immediately, | |
| all those bytes will be read (up to the buffer size). So, if the | |
| buffer size is 1 megabyte and there is 1 megabyte of data available | |
| to read, the buffer will be filled, regardless of timeout. | |
| This is a wrapper around os.read(). It uses select.select() or | |
| select.poll() to implement the timeout. ''' | |
| if self.closed: | |
| raise ValueError('I/O operation on closed file.') | |
| if self.use_poll: | |
| def select(timeout): | |
| return poll_ignore_interrupts([self.child_fd], timeout) | |
| else: | |
| def select(timeout): | |
| return select_ignore_interrupts([self.child_fd], [], [], timeout)[0] | |
| # If there is data available to read right now, read as much as | |
| # we can. We do this to increase performance if there are a lot | |
| # of bytes to be read. This also avoids calling isalive() too | |
| # often. See also: | |
| # * https://github.com/pexpect/pexpect/pull/304 | |
| # * http://trac.sagemath.org/ticket/10295 | |
| if select(0): | |
| try: | |
| incoming = super(spawn, self).read_nonblocking(size) | |
| except EOF: | |
| # Maybe the child is dead: update some attributes in that case | |
| self.isalive() | |
| raise | |
| while len(incoming) < size and select(0): | |
| try: | |
| incoming += super(spawn, self).read_nonblocking(size - len(incoming)) | |
| except EOF: | |
| # Maybe the child is dead: update some attributes in that case | |
| self.isalive() | |
| # Don't raise EOF, just return what we read so far. | |
| return incoming | |
| return incoming | |
| if timeout == -1: | |
| timeout = self.timeout | |
| if not self.isalive(): | |
| # The process is dead, but there may or may not be data | |
| # available to read. Note that some systems such as Solaris | |
| # do not give an EOF when the child dies. In fact, you can | |
| # still try to read from the child_fd -- it will block | |
| # forever or until TIMEOUT. For that reason, it's important | |
| # to do this check before calling select() with timeout. | |
| if select(0): | |
| return super(spawn, self).read_nonblocking(size) | |
| self.flag_eof = True | |
| raise EOF('End Of File (EOF). Braindead platform.') | |
| elif self.__irix_hack: | |
| # Irix takes a long time before it realizes a child was terminated. | |
| # Make sure that the timeout is at least 2 seconds. | |
| # FIXME So does this mean Irix systems are forced to always have | |
| # FIXME a 2 second delay when calling read_nonblocking? That sucks. | |
| if timeout is not None and timeout < 2: | |
| timeout = 2 | |
| # Because of the select(0) check above, we know that no data | |
| # is available right now. But if a non-zero timeout is given | |
| # (possibly timeout=None), we call select() with a timeout. | |
| if (timeout != 0) and select(timeout): | |
| return super(spawn, self).read_nonblocking(size) | |
| if not self.isalive(): | |
| # Some platforms, such as Irix, will claim that their | |
| # processes are alive; timeout on the select; and | |
| # then finally admit that they are not alive. | |
| self.flag_eof = True | |
| raise EOF('End of File (EOF). Very slow platform.') | |
| else: | |
| raise TIMEOUT('Timeout exceeded.') | |
| def write(self, s): | |
| '''This is similar to send() except that there is no return value. | |
| ''' | |
| self.send(s) | |
| def writelines(self, sequence): | |
| '''This calls write() for each element in the sequence. The sequence | |
| can be any iterable object producing strings, typically a list of | |
| strings. This does not add line separators. There is no return value. | |
| ''' | |
| for s in sequence: | |
| self.write(s) | |
| def send(self, s): | |
| '''Sends string ``s`` to the child process, returning the number of | |
| bytes written. If a logfile is specified, a copy is written to that | |
| log. | |
| The default terminal input mode is canonical processing unless set | |
| otherwise by the child process. This allows backspace and other line | |
| processing to be performed prior to transmitting to the receiving | |
| program. As this is buffered, there is a limited size of such buffer. | |
| On Linux systems, this is 4096 (defined by N_TTY_BUF_SIZE). All | |
| other systems honor the POSIX.1 definition PC_MAX_CANON -- 1024 | |
| on OSX, 256 on OpenSolaris, and 1920 on FreeBSD. | |
| This value may be discovered using fpathconf(3):: | |
| >>> from os import fpathconf | |
| >>> print(fpathconf(0, 'PC_MAX_CANON')) | |
| 256 | |
| On such a system, only 256 bytes may be received per line. Any | |
| subsequent bytes received will be discarded. BEL (``'\a'``) is then | |
| sent to output if IMAXBEL (termios.h) is set by the tty driver. | |
| This is usually enabled by default. Linux does not honor this as | |
| an option -- it behaves as though it is always set on. | |
| Canonical input processing may be disabled altogether by executing | |
| a shell, then stty(1), before executing the final program:: | |
| >>> bash = pexpect.spawn('/bin/bash', echo=False) | |
| >>> bash.sendline('stty -icanon') | |
| >>> bash.sendline('base64') | |
| >>> bash.sendline('x' * 5000) | |
| ''' | |
| if self.delaybeforesend is not None: | |
| time.sleep(self.delaybeforesend) | |
| s = self._coerce_send_string(s) | |
| self._log(s, 'send') | |
| b = self._encoder.encode(s, final=False) | |
| return os.write(self.child_fd, b) | |
| def sendline(self, s=''): | |
| '''Wraps send(), sending string ``s`` to child process, with | |
| ``os.linesep`` automatically appended. Returns number of bytes | |
| written. Only a limited number of bytes may be sent for each | |
| line in the default terminal mode, see docstring of :meth:`send`. | |
| ''' | |
| s = self._coerce_send_string(s) | |
| return self.send(s + self.linesep) | |
| def _log_control(self, s): | |
| """Write control characters to the appropriate log files""" | |
| if self.encoding is not None: | |
| s = s.decode(self.encoding, 'replace') | |
| self._log(s, 'send') | |
| def sendcontrol(self, char): | |
| '''Helper method that wraps send() with mnemonic access for sending control | |
| character to the child (such as Ctrl-C or Ctrl-D). For example, to send | |
| Ctrl-G (ASCII 7, bell, '\a'):: | |
| child.sendcontrol('g') | |
| See also, sendintr() and sendeof(). | |
| ''' | |
| n, byte = self.ptyproc.sendcontrol(char) | |
| self._log_control(byte) | |
| return n | |
| def sendeof(self): | |
| '''This sends an EOF to the child. This sends a character which causes | |
| the pending parent output buffer to be sent to the waiting child | |
| program without waiting for end-of-line. If it is the first character | |
| of the line, the read() in the user program returns 0, which signifies | |
| end-of-file. This means to work as expected a sendeof() has to be | |
| called at the beginning of a line. This method does not send a newline. | |
| It is the responsibility of the caller to ensure the eof is sent at the | |
| beginning of a line. ''' | |
| n, byte = self.ptyproc.sendeof() | |
| self._log_control(byte) | |
| def sendintr(self): | |
| '''This sends a SIGINT to the child. It does not require | |
| the SIGINT to be the first character on a line. ''' | |
| n, byte = self.ptyproc.sendintr() | |
| self._log_control(byte) | |
| def flag_eof(self): | |
| return self.ptyproc.flag_eof | |
| def flag_eof(self, value): | |
| self.ptyproc.flag_eof = value | |
| def eof(self): | |
| '''This returns True if the EOF exception was ever raised. | |
| ''' | |
| return self.flag_eof | |
| def terminate(self, force=False): | |
| '''This forces a child process to terminate. It starts nicely with | |
| SIGHUP and SIGINT. If "force" is True then moves onto SIGKILL. This | |
| returns True if the child was terminated. This returns False if the | |
| child could not be terminated. ''' | |
| if not self.isalive(): | |
| return True | |
| try: | |
| self.kill(signal.SIGHUP) | |
| time.sleep(self.delayafterterminate) | |
| if not self.isalive(): | |
| return True | |
| self.kill(signal.SIGCONT) | |
| time.sleep(self.delayafterterminate) | |
| if not self.isalive(): | |
| return True | |
| self.kill(signal.SIGINT) | |
| time.sleep(self.delayafterterminate) | |
| if not self.isalive(): | |
| return True | |
| if force: | |
| self.kill(signal.SIGKILL) | |
| time.sleep(self.delayafterterminate) | |
| if not self.isalive(): | |
| return True | |
| else: | |
| return False | |
| return False | |
| except OSError: | |
| # I think there are kernel timing issues that sometimes cause | |
| # this to happen. I think isalive() reports True, but the | |
| # process is dead to the kernel. | |
| # Make one last attempt to see if the kernel is up to date. | |
| time.sleep(self.delayafterterminate) | |
| if not self.isalive(): | |
| return True | |
| else: | |
| return False | |
| def wait(self): | |
| '''This waits until the child exits. This is a blocking call. This will | |
| not read any data from the child, so this will block forever if the | |
| child has unread output and has terminated. In other words, the child | |
| may have printed output then called exit(), but, the child is | |
| technically still alive until its output is read by the parent. | |
| This method is non-blocking if :meth:`wait` has already been called | |
| previously or :meth:`isalive` method returns False. It simply returns | |
| the previously determined exit status. | |
| ''' | |
| ptyproc = self.ptyproc | |
| with _wrap_ptyprocess_err(): | |
| # exception may occur if "Is some other process attempting | |
| # "job control with our child pid?" | |
| exitstatus = ptyproc.wait() | |
| self.status = ptyproc.status | |
| self.exitstatus = ptyproc.exitstatus | |
| self.signalstatus = ptyproc.signalstatus | |
| self.terminated = True | |
| return exitstatus | |
| def isalive(self): | |
| '''This tests if the child process is running or not. This is | |
| non-blocking. If the child was terminated then this will read the | |
| exitstatus or signalstatus of the child. This returns True if the child | |
| process appears to be running or False if not. It can take literally | |
| SECONDS for Solaris to return the right status. ''' | |
| ptyproc = self.ptyproc | |
| with _wrap_ptyprocess_err(): | |
| alive = ptyproc.isalive() | |
| if not alive: | |
| self.status = ptyproc.status | |
| self.exitstatus = ptyproc.exitstatus | |
| self.signalstatus = ptyproc.signalstatus | |
| self.terminated = True | |
| return alive | |
| def kill(self, sig): | |
| '''This sends the given signal to the child application. In keeping | |
| with UNIX tradition it has a misleading name. It does not necessarily | |
| kill the child unless you send the right signal. ''' | |
| # Same as os.kill, but the pid is given for you. | |
| if self.isalive(): | |
| os.kill(self.pid, sig) | |
| def getwinsize(self): | |
| '''This returns the terminal window size of the child tty. The return | |
| value is a tuple of (rows, cols). ''' | |
| return self.ptyproc.getwinsize() | |
| def setwinsize(self, rows, cols): | |
| '''This sets the terminal window size of the child tty. This will cause | |
| a SIGWINCH signal to be sent to the child. This does not change the | |
| physical window size. It changes the size reported to TTY-aware | |
| applications like vi or curses -- applications that respond to the | |
| SIGWINCH signal. ''' | |
| return self.ptyproc.setwinsize(rows, cols) | |
| def interact(self, escape_character=chr(29), | |
| input_filter=None, output_filter=None): | |
| '''This gives control of the child process to the interactive user (the | |
| human at the keyboard). Keystrokes are sent to the child process, and | |
| the stdout and stderr output of the child process is printed. This | |
| simply echos the child stdout and child stderr to the real stdout and | |
| it echos the real stdin to the child stdin. When the user types the | |
| escape_character this method will return None. The escape_character | |
| will not be transmitted. The default for escape_character is | |
| entered as ``Ctrl - ]``, the very same as BSD telnet. To prevent | |
| escaping, escape_character may be set to None. | |
| If a logfile is specified, then the data sent and received from the | |
| child process in interact mode is duplicated to the given log. | |
| You may pass in optional input and output filter functions. These | |
| functions should take bytes array and return bytes array too. Even | |
| with ``encoding='utf-8'`` support, meth:`interact` will always pass | |
| input_filter and output_filter bytes. You may need to wrap your | |
| function to decode and encode back to UTF-8. | |
| The output_filter will be passed all the output from the child process. | |
| The input_filter will be passed all the keyboard input from the user. | |
| The input_filter is run BEFORE the check for the escape_character. | |
| Note that if you change the window size of the parent the SIGWINCH | |
| signal will not be passed through to the child. If you want the child | |
| window size to change when the parent's window size changes then do | |
| something like the following example:: | |
| import pexpect, struct, fcntl, termios, signal, sys | |
| def sigwinch_passthrough (sig, data): | |
| s = struct.pack("HHHH", 0, 0, 0, 0) | |
| a = struct.unpack('hhhh', fcntl.ioctl(sys.stdout.fileno(), | |
| termios.TIOCGWINSZ , s)) | |
| if not p.closed: | |
| p.setwinsize(a[0],a[1]) | |
| # Note this 'p' is global and used in sigwinch_passthrough. | |
| p = pexpect.spawn('/bin/bash') | |
| signal.signal(signal.SIGWINCH, sigwinch_passthrough) | |
| p.interact() | |
| ''' | |
| # Flush the buffer. | |
| self.write_to_stdout(self.buffer) | |
| self.stdout.flush() | |
| self._buffer = self.buffer_type() | |
| mode = tty.tcgetattr(self.STDIN_FILENO) | |
| tty.setraw(self.STDIN_FILENO) | |
| if escape_character is not None and PY3: | |
| escape_character = escape_character.encode('latin-1') | |
| try: | |
| self.__interact_copy(escape_character, input_filter, output_filter) | |
| finally: | |
| tty.tcsetattr(self.STDIN_FILENO, tty.TCSAFLUSH, mode) | |
| def __interact_writen(self, fd, data): | |
| '''This is used by the interact() method. | |
| ''' | |
| while data != b'' and self.isalive(): | |
| n = os.write(fd, data) | |
| data = data[n:] | |
| def __interact_read(self, fd): | |
| '''This is used by the interact() method. | |
| ''' | |
| return os.read(fd, 1000) | |
| def __interact_copy( | |
| self, escape_character=None, input_filter=None, output_filter=None | |
| ): | |
| '''This is used by the interact() method. | |
| ''' | |
| while self.isalive(): | |
| if self.use_poll: | |
| r = poll_ignore_interrupts([self.child_fd, self.STDIN_FILENO]) | |
| else: | |
| r, w, e = select_ignore_interrupts( | |
| [self.child_fd, self.STDIN_FILENO], [], [] | |
| ) | |
| if self.child_fd in r: | |
| try: | |
| data = self.__interact_read(self.child_fd) | |
| except OSError as err: | |
| if err.args[0] == errno.EIO: | |
| # Linux-style EOF | |
| break | |
| raise | |
| if data == b'': | |
| # BSD-style EOF | |
| break | |
| if output_filter: | |
| data = output_filter(data) | |
| self._log(data, 'read') | |
| os.write(self.STDOUT_FILENO, data) | |
| if self.STDIN_FILENO in r: | |
| data = self.__interact_read(self.STDIN_FILENO) | |
| if input_filter: | |
| data = input_filter(data) | |
| i = -1 | |
| if escape_character is not None: | |
| i = data.rfind(escape_character) | |
| if i != -1: | |
| data = data[:i] | |
| if data: | |
| self._log(data, 'send') | |
| self.__interact_writen(self.child_fd, data) | |
| break | |
| self._log(data, 'send') | |
| self.__interact_writen(self.child_fd, data) | |
| def spawnu(*args, **kwargs): | |
| """Deprecated: pass encoding to spawn() instead.""" | |
| kwargs.setdefault('encoding', 'utf-8') | |
| return spawn(*args, **kwargs) | |
Xet Storage Details
- Size:
- 37.4 kB
- Xet hash:
- 49737b3a8e926762433636fb237822a7674a850604dba0d0126f2896c483be6f
·
Xet efficiently stores files, intelligently splitting them into unique chunks and accelerating uploads and downloads. More info.