io.py
151 lines
| 4.5 KiB
| text/x-python
|
PythonLexer
Brian Granger
|
r2498 | # encoding: utf-8 | ||
""" | ||||
IO related utilities. | ||||
""" | ||||
Min RK
|
r21111 | # Copyright (c) IPython Development Team. | ||
# Distributed under the terms of the Modified BSD License. | ||||
Brian Granger
|
r2498 | |||
Min RK
|
r21111 | |||
John Kirkham
|
r21762 | import atexit | ||
Brandon Parsons
|
r6653 | import os | ||
Brian Granger
|
r2498 | import sys | ||
import tempfile | ||||
Samreen Zarroug
|
r26154 | from pathlib import Path | ||
Matthias Bussonnier
|
r21298 | from warnings import warn | ||
Thomas Kluyver
|
r22192 | |||
from IPython.utils.decorators import undoc | ||||
MinRK
|
r12223 | from .capture import CapturedIO, capture_output | ||
Brian Granger
|
r2498 | |||
class Tee(object): | ||||
"""A class to duplicate an output stream to stdout/err. | ||||
This works in a manner very similar to the Unix 'tee' command. | ||||
When the object is closed or deleted, it closes the original file given to | ||||
it for duplication. | ||||
""" | ||||
# Inspired by: | ||||
# http://mail.python.org/pipermail/python-list/2007-May/442737.html | ||||
Thomas Kluyver
|
r4763 | def __init__(self, file_or_name, mode="w", channel='stdout'): | ||
Brian Granger
|
r2498 | """Construct a new Tee object. | ||
Parameters | ||||
---------- | ||||
file_or_name : filename or open filehandle (writable) | ||||
Matthias Bussonnier
|
r26419 | File that will be duplicated | ||
Brian Granger
|
r2498 | mode : optional, valid mode for open(). | ||
Matthias Bussonnier
|
r26419 | If a filename was give, open with this mode. | ||
Bernardo B. Marques
|
r4872 | channel : str, one of ['stdout', 'stderr'] | ||
Brian Granger
|
r2498 | """ | ||
if channel not in ['stdout', 'stderr']: | ||||
raise ValueError('Invalid channel spec %s' % channel) | ||||
Bernardo B. Marques
|
r4872 | |||
Thomas Kluyver
|
r4763 | if hasattr(file_or_name, 'write') and hasattr(file_or_name, 'seek'): | ||
Brian Granger
|
r2498 | self.file = file_or_name | ||
else: | ||||
gousaiyang
|
r27495 | encoding = None if "b" in mode else "utf-8" | ||
gousaiyang
|
r27494 | self.file = open(file_or_name, mode, encoding=encoding) | ||
Brian Granger
|
r2498 | self.channel = channel | ||
self.ostream = getattr(sys, channel) | ||||
setattr(sys, channel, self) | ||||
self._closed = False | ||||
def close(self): | ||||
"""Close the file and restore the channel.""" | ||||
self.flush() | ||||
setattr(sys, self.channel, self.ostream) | ||||
self.file.close() | ||||
self._closed = True | ||||
def write(self, data): | ||||
"""Write data to both channels.""" | ||||
self.file.write(data) | ||||
self.ostream.write(data) | ||||
self.ostream.flush() | ||||
def flush(self): | ||||
"""Flush both channels.""" | ||||
self.file.flush() | ||||
self.ostream.flush() | ||||
def __del__(self): | ||||
if not self._closed: | ||||
self.close() | ||||
Paul Ivanov
|
r13609 | def ask_yes_no(prompt, default=None, interrupt=None): | ||
Brian Granger
|
r2498 | """Asks a question and returns a boolean (y/n) answer. | ||
If default is given (one of 'y','n'), it is used if the user input is | ||||
Paul Ivanov
|
r13609 | empty. If interrupt is given (one of 'y','n'), it is used if the user | ||
presses Ctrl-C. Otherwise the question is repeated until an answer is | ||||
given. | ||||
Brian Granger
|
r2498 | |||
An EOF is treated as the default answer. If there is no default, an | ||||
exception is raised to prevent infinite loops. | ||||
Valid answers are: y/yes/n/no (match is not case sensitive).""" | ||||
answers = {'y':True,'n':False,'yes':True,'no':False} | ||||
ans = None | ||||
while ans not in answers.keys(): | ||||
try: | ||||
Thomas Kluyver
|
r13355 | ans = input(prompt+' ').lower() | ||
Brian Granger
|
r2498 | if not ans: # response was an empty string | ||
ans = default | ||||
except KeyboardInterrupt: | ||||
Paul Ivanov
|
r13609 | if interrupt: | ||
ans = interrupt | ||||
sachet-mittal
|
r23031 | print("\r") | ||
Brian Granger
|
r2498 | except EOFError: | ||
if default in answers.keys(): | ||||
ans = default | ||||
Thomas Spura
|
r4430 | print() | ||
Brian Granger
|
r2498 | else: | ||
raise | ||||
return answers[ans] | ||||
def temp_pyfile(src, ext='.py'): | ||||
"""Make a temporary python file, return filename and filehandle. | ||||
Parameters | ||||
---------- | ||||
src : string or list of strings (no need for ending newlines if list) | ||||
Matthias Bussonnier
|
r26419 | Source code to be written to the file. | ||
Brian Granger
|
r2498 | ext : optional, string | ||
Matthias Bussonnier
|
r26419 | Extension for the generated file. | ||
Brian Granger
|
r2498 | |||
Returns | ||||
------- | ||||
(filename, open filehandle) | ||||
Matthias Bussonnier
|
r26419 | It is the caller's responsibility to close the open file and unlink it. | ||
Brian Granger
|
r2498 | """ | ||
fname = tempfile.mkstemp(ext)[1] | ||||
gousaiyang
|
r27495 | with open(Path(fname), "w", encoding="utf-8") as f: | ||
Matthias Bussonnier
|
r25106 | f.write(src) | ||
f.flush() | ||||
return fname | ||||
Brian Granger
|
r2498 | |||
Matthias Bussonnier
|
r24275 | @undoc | ||
Fernando Perez
|
r2874 | def raw_print(*args, **kw): | ||
Matthias Bussonnier
|
r24275 | """DEPRECATED: Raw print to sys.__stdout__, otherwise identical interface to print().""" | ||
warn("IPython.utils.io.raw_print has been deprecated since IPython 7.0", DeprecationWarning, stacklevel=2) | ||||
Bernardo B. Marques
|
r4872 | |||
Fernando Perez
|
r2856 | print(*args, sep=kw.get('sep', ' '), end=kw.get('end', '\n'), | ||
file=sys.__stdout__) | ||||
sys.__stdout__.flush() | ||||
Matthias Bussonnier
|
r24275 | @undoc | ||
Fernando Perez
|
r2874 | def raw_print_err(*args, **kw): | ||
Matthias Bussonnier
|
r24275 | """DEPRECATED: Raw print to sys.__stderr__, otherwise identical interface to print().""" | ||
warn("IPython.utils.io.raw_print_err has been deprecated since IPython 7.0", DeprecationWarning, stacklevel=2) | ||||
Bernardo B. Marques
|
r4872 | |||
Fernando Perez
|
r2856 | print(*args, sep=kw.get('sep', ' '), end=kw.get('end', '\n'), | ||
file=sys.__stderr__) | ||||
Fernando Perez
|
r2838 | sys.__stderr__.flush() | ||