##// END OF EJS Templates
Refuse to install event loop hooks when not using `prompt_toolkit` (#14132)...
Refuse to install event loop hooks when not using `prompt_toolkit` (#14132) Without this, `%gui` is effectively a no-op but the user thinks it works. For example. If running `ipython`: ``` In [1]: import matplotlib; matplotlib.use('QtAgg'); from matplotlib import pyplot; pyplot.ion(); pyplot.plot([1, 2, 3, 4]) Installed qt6 event loop hook. Out[1]: [<matplotlib.lines.Line2D at 0x1ba2f59d2a0>] ``` The window appears and responds as expected. If running `ipython --simple-prompt`, the user would see the same output, when in fact no event loop hook was installed since it's not supported without `prompt_toolkit`. The resulting Qt window is unresponsive because the event loop is not running, i.e. with `--simple-prompt`, Qt windows should block (but `pyplot` doesn't/can't know to do that) With this PR, the user will see: ``` In [1]: import matplotlib; matplotlib.use('QtAgg'); from matplotlib import pyplot; pyplot.ion(); pyplot.plot([1, 2, 3, 4]) Cannot install event loop hook for "qt" when running with `--simple-prompt`. NOTE: Tk is supported natively; use Tk apps and Tk backends with `--simple-prompt`. Out[1]: [<matplotlib.lines.Line2D at 0x170be0c0310>] ``` They'll still get an unresponsive Qt window, but they'll at least be told this can't work (while anything using Tk will work just fine).

File last commit:

r23969:d73e5f74
r28368:22fc5ab5 merge
Show More
capture.py
170 lines | 5.0 KiB | text/x-python | PythonLexer
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 # encoding: utf-8
MinRK
remove `source` key from display_data
r16585 """IO capturing utilities."""
MinRK
capture rich output as well as stdout/err in capture_output...
r12223
MinRK
remove `source` key from display_data
r16585 # Copyright (c) IPython Development Team.
# Distributed under the terms of the Modified BSD License.
MinRK
capture rich output as well as stdout/err in capture_output...
r12223
import sys
Srinivas Reddy Thatiparthy
remove PY3 var
r23111 from io import StringIO
MinRK
capture rich output as well as stdout/err in capture_output...
r12223
#-----------------------------------------------------------------------------
# Classes and functions
#-----------------------------------------------------------------------------
class RichOutput(object):
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969 def __init__(self, data=None, metadata=None, transient=None, update=False):
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 self.data = data or {}
self.metadata = metadata or {}
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969 self.transient = transient or {}
self.update = update
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 def display(self):
from IPython.display import publish_display_data
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969 publish_display_data(data=self.data, metadata=self.metadata,
transient=self.transient, update=self.update)
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 def _repr_mime_(self, mime):
if mime not in self.data:
return
data = self.data[mime]
if mime in self.metadata:
return data, self.metadata[mime]
else:
return data
Min RK
implement GIF support without registering a new formatter...
r23849
def _repr_mimebundle_(self, include=None, exclude=None):
return self.data, self.metadata
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 def _repr_html_(self):
return self._repr_mime_("text/html")
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 def _repr_latex_(self):
return self._repr_mime_("text/latex")
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 def _repr_json_(self):
return self._repr_mime_("application/json")
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 def _repr_javascript_(self):
return self._repr_mime_("application/javascript")
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 def _repr_png_(self):
return self._repr_mime_("image/png")
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 def _repr_jpeg_(self):
MinRK
default values for RichOutput attributes
r12227 return self._repr_mime_("image/jpeg")
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 def _repr_svg_(self):
return self._repr_mime_("image/svg+xml")
class CapturedIO(object):
Paul Ivanov
add --no-display flag to %%capture...
r12225 """Simple object for containing captured stdout/err and rich display StringIO objects
Each instance `c` has three attributes:
Thomas Kluyver
Improvements to docs formatting.
r12553 - ``c.stdout`` : standard output as a string
- ``c.stderr`` : standard error as a string
- ``c.outputs``: a list of rich display outputs
Paul Ivanov
add --no-display flag to %%capture...
r12225
Thomas Kluyver
Improvements to docs formatting.
r12553 Additionally, there's a ``c.show()`` method which will print all of the
above in the same order, and can be invoked simply via ``c()``.
Paul Ivanov
add --no-display flag to %%capture...
r12225 """
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 def __init__(self, stdout, stderr, outputs=None):
self._stdout = stdout
self._stderr = stderr
if outputs is None:
outputs = []
self._outputs = outputs
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 def __str__(self):
return self.stdout
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 @property
def stdout(self):
Paul Ivanov
updated Basic Output notebook with new %%capture
r12226 "Captured standard output"
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 if not self._stdout:
return ''
return self._stdout.getvalue()
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 @property
def stderr(self):
Paul Ivanov
updated Basic Output notebook with new %%capture
r12226 "Captured standard error"
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 if not self._stderr:
return ''
return self._stderr.getvalue()
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969
MinRK
add outputs property on CapturedIO
r12224 @property
def outputs(self):
Paul Ivanov
updated Basic Output notebook with new %%capture
r12226 """A list of the captured rich display outputs, if any.
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969
Thomas Kluyver
Improvements to docs formatting.
r12553 If you have a CapturedIO object ``c``, these can be displayed in IPython
using::
Paul Ivanov
updated Basic Output notebook with new %%capture
r12226
from IPython.display import display
for o in c.outputs:
display(o)
"""
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969 return [ RichOutput(**kargs) for kargs in self._outputs ]
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 def show(self):
"""write my output to sys.stdout/err as appropriate"""
sys.stdout.write(self.stdout)
sys.stderr.write(self.stderr)
sys.stdout.flush()
sys.stderr.flush()
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969 for kargs in self._outputs:
RichOutput(**kargs).display()
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 __call__ = show
class capture_output(object):
"""context manager for capturing stdout/err"""
stdout = True
stderr = True
display = True
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 def __init__(self, stdout=True, stderr=True, display=True):
self.stdout = stdout
self.stderr = stderr
self.display = display
self.shell = None
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 def __enter__(self):
from IPython.core.getipython import get_ipython
from IPython.core.displaypub import CapturingDisplayPublisher
Thomas Kluyver
Also capture execution results using sys.displayhook...
r22774 from IPython.core.displayhook import CapturingDisplayHook
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 self.sys_stdout = sys.stdout
self.sys_stderr = sys.stderr
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 if self.display:
self.shell = get_ipython()
if self.shell is None:
self.save_display_pub = None
self.display = False
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969
Paul Ivanov
add --no-display flag to %%capture...
r12225 stdout = stderr = outputs = None
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 if self.stdout:
stdout = sys.stdout = StringIO()
if self.stderr:
stderr = sys.stderr = StringIO()
if self.display:
self.save_display_pub = self.shell.display_pub
self.shell.display_pub = CapturingDisplayPublisher()
outputs = self.shell.display_pub.outputs
Thomas Kluyver
Also capture execution results using sys.displayhook...
r22774 self.save_display_hook = sys.displayhook
sys.displayhook = CapturingDisplayHook(shell=self.shell,
outputs=outputs)
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 return CapturedIO(stdout, stderr, outputs)
Henry Fredrick Schreiner
Adding transient and update, change to dict for outputs
r23969
MinRK
capture rich output as well as stdout/err in capture_output...
r12223 def __exit__(self, exc_type, exc_value, traceback):
sys.stdout = self.sys_stdout
sys.stderr = self.sys_stderr
if self.display and self.shell:
self.shell.display_pub = self.save_display_pub
Thomas Kluyver
Also capture execution results using sys.displayhook...
r22774 sys.displayhook = self.save_display_hook