ipapp.py
338 lines
| 12.1 KiB
| text/x-python
|
PythonLexer
Brian Granger
|
r2202 | # encoding: utf-8 | ||
""" | ||||
Matthias Bussonnier
|
r27616 | The :class:`~traitlets.config.application.Application` object for the command | ||
Brian Granger
|
r2301 | line :command:`ipython` program. | ||
Brian Granger
|
r2202 | """ | ||
Min RK
|
r20890 | # Copyright (c) IPython Development Team. | ||
# Distributed under the terms of the Modified BSD License. | ||||
Brian Granger
|
r2506 | |||
Brian Granger
|
r2202 | |||
Brian Granger
|
r2252 | import logging | ||
Brian Granger
|
r2203 | import os | ||
import sys | ||||
Min RK
|
r22340 | import warnings | ||
Brian Granger
|
r2203 | |||
Min RK
|
r21253 | from traitlets.config.loader import Config | ||
Thomas Kluyver
|
r23932 | from traitlets.config.application import boolean_flag, catch_config_error | ||
Brian Granger
|
r2506 | from IPython.core import release | ||
MinRK
|
r3963 | from IPython.core import usage | ||
MinRK
|
r5231 | from IPython.core.completer import IPCompleter | ||
Brian Granger
|
r2506 | from IPython.core.crashhandler import CrashHandler | ||
MinRK
|
r3963 | from IPython.core.formatters import PlainTextFormatter | ||
MinRK
|
r6823 | from IPython.core.history import HistoryManager | ||
MinRK
|
r4023 | from IPython.core.application import ( | ||
MinRK
|
r3963 | ProfileDir, BaseIPythonApplication, base_flags, base_aliases | ||
Brian Granger
|
r2245 | ) | ||
Matthias Bussonnier
|
r27503 | from IPython.core.magic import MagicsManager | ||
Joshua Storck
|
r23624 | from IPython.core.magics import ( | ||
ScriptMagics, LoggingMagics | ||||
) | ||||
MinRK
|
r3968 | from IPython.core.shellapp import ( | ||
InteractiveShellApp, shell_flags, shell_aliases | ||||
) | ||||
Thomas Kluyver
|
r12331 | from IPython.extensions.storemagic import StoreMagics | ||
Min RK
|
r22549 | from .interactiveshell import TerminalInteractiveShell | ||
Min RK
|
r21253 | from IPython.paths import get_ipython_dir | ||
from traitlets import ( | ||||
Thomas Kluyver
|
r23932 | Bool, List, default, observe, Type | ||
MinRK
|
r3963 | ) | ||
Brian Granger
|
r2203 | |||
#----------------------------------------------------------------------------- | ||||
Fernando Perez
|
r2427 | # Globals, utilities and helpers | ||
Brian Granger
|
r2203 | #----------------------------------------------------------------------------- | ||
Brian Granger
|
r4216 | _examples = """ | ||
Paul Ivanov
|
r12015 | ipython --matplotlib # enable matplotlib integration | ||
Bing Xia
|
r12154 | ipython --matplotlib=qt # enable matplotlib integration with qt4 backend | ||
Paul Ivanov
|
r12015 | |||
Brian E. Granger
|
r4219 | ipython --log-level=DEBUG # set logging to DEBUG | ||
Brian Granger
|
r4216 | ipython --profile=foo # start with profile foo | ||
Brian E. Granger
|
r4218 | |||
MinRK
|
r6169 | ipython profile create foo # create profile foo w/ default config files | ||
ipython help profile # show the help for the profile subcmd | ||||
MinRK
|
r6901 | |||
ipython locate # print the path to the IPython directory | ||||
ipython locate profile foo # print the path to the directory for profile `foo` | ||||
Brian Granger
|
r4216 | """ | ||
Brian Granger
|
r2501 | |||
Fernando Perez
|
r2427 | #----------------------------------------------------------------------------- | ||
Brian Granger
|
r2506 | # Crash handler for this application | ||
#----------------------------------------------------------------------------- | ||||
class IPAppCrashHandler(CrashHandler): | ||||
"""sys.excepthook for IPython itself, leaves a detailed report on disk.""" | ||||
def __init__(self, app): | ||||
MinRK
|
r9188 | contact_name = release.author | ||
MinRK
|
r5316 | contact_email = release.author_email | ||
bug_tracker = 'https://github.com/ipython/ipython/issues' | ||||
Brian Granger
|
r2506 | super(IPAppCrashHandler,self).__init__( | ||
app, contact_name, contact_email, bug_tracker | ||||
) | ||||
def make_report(self,traceback): | ||||
"""Return a string containing a crash report.""" | ||||
sec_sep = self.section_sep | ||||
# Start with parent report | ||||
report = [super(IPAppCrashHandler, self).make_report(traceback)] | ||||
# Add interactive-specific info we may have | ||||
rpt_add = report.append | ||||
try: | ||||
rpt_add(sec_sep+"History of session input:") | ||||
for line in self.app.shell.user_ns['_ih']: | ||||
rpt_add(line) | ||||
rpt_add('\n*** Last line of input (may not be in above history):\n') | ||||
rpt_add(self.app.shell._last_input_line+'\n') | ||||
except: | ||||
pass | ||||
return ''.join(report) | ||||
MinRK
|
r3963 | #----------------------------------------------------------------------------- | ||
# Aliases and Flags | ||||
#----------------------------------------------------------------------------- | ||||
flags = dict(base_flags) | ||||
MinRK
|
r3968 | flags.update(shell_flags) | ||
MinRK
|
r5610 | frontend_flags = {} | ||
addflag = lambda *args: frontend_flags.update(boolean_flag(*args)) | ||||
MinRK
|
r3963 | addflag('autoedit-syntax', 'TerminalInteractiveShell.autoedit_syntax', | ||
'Turn on auto editing of files with syntax errors.', | ||||
'Turn off auto editing of files with syntax errors.' | ||||
) | ||||
Matthias Bussonnier
|
r22416 | addflag('simple-prompt', 'TerminalInteractiveShell.simple_prompt', | ||
"Force simple minimal prompt using `raw_input`", | ||||
"Use a rich interactive prompt with prompt_toolkit", | ||||
) | ||||
MinRK
|
r3968 | addflag('banner', 'TerminalIPythonApp.display_banner', | ||
MinRK
|
r3963 | "Display a banner upon starting IPython.", | ||
"Don't display a banner upon starting IPython." | ||||
) | ||||
addflag('confirm-exit', 'TerminalInteractiveShell.confirm_exit', | ||||
"""Set to confirm when you try to exit IPython with an EOF (Control-D | ||||
MinRK
|
r3967 | in Unix, Control-Z/Enter in Windows). By typing 'exit' or 'quit', | ||
you can force a direct exit without any confirmation.""", | ||||
MinRK
|
r3963 | "Don't prompt the user when exiting." | ||
) | ||||
addflag('term-title', 'TerminalInteractiveShell.term_title', | ||||
"Enable auto setting the terminal title.", | ||||
"Disable auto setting the terminal title." | ||||
) | ||||
classic_config = Config() | ||||
classic_config.InteractiveShell.cache_size = 0 | ||||
classic_config.PlainTextFormatter.pprint = False | ||||
Matthias Bussonnier
|
r22447 | classic_config.TerminalInteractiveShell.prompts_class='IPython.terminal.prompts.ClassicPrompts' | ||
MinRK
|
r3963 | classic_config.InteractiveShell.separate_in = '' | ||
classic_config.InteractiveShell.separate_out = '' | ||||
classic_config.InteractiveShell.separate_out2 = '' | ||||
classic_config.InteractiveShell.colors = 'NoColor' | ||||
classic_config.InteractiveShell.xmode = 'Plain' | ||||
MinRK
|
r5610 | frontend_flags['classic']=( | ||
MinRK
|
r3963 | classic_config, | ||
"Gives IPython a similar feel to the classic Python prompt." | ||||
) | ||||
# # log doesn't make so much sense this way anymore | ||||
# paa('--log','-l', | ||||
# action='store_true', dest='InteractiveShell.logstart', | ||||
# help="Start logging to the default log file (./ipython_log.py).") | ||||
# | ||||
# # quick is harder to implement | ||||
MinRK
|
r5610 | frontend_flags['quick']=( | ||
MinRK
|
r3968 | {'TerminalIPythonApp' : {'quick' : True}}, | ||
MinRK
|
r3963 | "Enable quick startup with no config files." | ||
) | ||||
MinRK
|
r5610 | frontend_flags['i'] = ( | ||
MinRK
|
r3968 | {'TerminalIPythonApp' : {'force_interact' : True}}, | ||
Joel Nothman
|
r21556 | """If running code from the command line, become interactive afterwards. | ||
It is often useful to follow this with `--` to treat remaining flags as | ||||
script arguments. | ||||
""" | ||||
MinRK
|
r3963 | ) | ||
MinRK
|
r5610 | flags.update(frontend_flags) | ||
MinRK
|
r3963 | |||
aliases = dict(base_aliases) | ||||
Matthias Bussonnier
|
r28028 | aliases.update(shell_aliases) # type: ignore[arg-type] | ||
MinRK
|
r3963 | |||
Brian Granger
|
r2506 | #----------------------------------------------------------------------------- | ||
Fernando Perez
|
r2427 | # Main classes and functions | ||
#----------------------------------------------------------------------------- | ||||
Brian Granger
|
r2245 | |||
MinRK
|
r6901 | |||
class LocateIPythonApp(BaseIPythonApplication): | ||||
description = """print the path to the IPython dir""" | ||||
Thomas Kluyver
|
r23932 | subcommands = dict( | ||
MinRK
|
r6901 | profile=('IPython.core.profileapp.ProfileLocate', | ||
"print the path to an IPython profile directory", | ||||
), | ||||
Thomas Kluyver
|
r23932 | ) | ||
MinRK
|
r6901 | def start(self): | ||
if self.subapp is not None: | ||||
return self.subapp.start() | ||||
else: | ||||
Thomas Kluyver
|
r13348 | print(self.ipython_dir) | ||
MinRK
|
r6901 | |||
MinRK
|
r3968 | class TerminalIPythonApp(BaseIPythonApplication, InteractiveShellApp): | ||
Matthias Bussonnier
|
r28565 | name = "ipython" | ||
MinRK
|
r3963 | description = usage.cl_usage | ||
Matthias Bussonnier
|
r28028 | crash_handler_class = IPAppCrashHandler # typing: ignore[assignment] | ||
Brian Granger
|
r4216 | examples = _examples | ||
Brian Granger
|
r4215 | |||
Thomas Kluyver
|
r23932 | flags = flags | ||
aliases = aliases | ||||
MinRK
|
r4462 | classes = List() | ||
Matthias Bussonnier
|
r23505 | |||
interactive_shell_class = Type( | ||||
klass=object, # use default_value otherwise which only allow subclasses. | ||||
default_value=TerminalInteractiveShell, | ||||
help="Class to use to instantiate the TerminalInteractiveShell object. Useful for custom Frontends" | ||||
).tag(config=True) | ||||
Min RK
|
r22340 | @default('classes') | ||
MinRK
|
r4462 | def _classes_default(self): | ||
"""This has to be in a method, for TerminalIPythonApp to be available.""" | ||||
return [ | ||||
Matthias Bussonnier
|
r28565 | InteractiveShellApp, # ShellApp comes before TerminalApp, because | ||
MinRK
|
r4462 | self.__class__, # it will also affect subclasses (e.g. QtConsole) | ||
TerminalInteractiveShell, | ||||
MinRK
|
r6823 | HistoryManager, | ||
Matthias Bussonnier
|
r27503 | MagicsManager, | ||
MinRK
|
r4462 | ProfileDir, | ||
PlainTextFormatter, | ||||
MinRK
|
r5231 | IPCompleter, | ||
MinRK
|
r7417 | ScriptMagics, | ||
Joshua Storck
|
r23624 | LoggingMagics, | ||
Thomas Kluyver
|
r12331 | StoreMagics, | ||
MinRK
|
r4462 | ] | ||
Bernardo B. Marques
|
r4872 | |||
Matthias Bussonnier
|
r21836 | subcommands = dict( | ||
profile = ("IPython.core.profileapp.ProfileApp", | ||||
"Create and manage IPython profiles." | ||||
), | ||||
kernel = ("ipykernel.kernelapp.IPKernelApp", | ||||
"Start a kernel without an attached frontend." | ||||
), | ||||
locate=('IPython.terminal.ipapp.LocateIPythonApp', | ||||
LocateIPythonApp.description | ||||
), | ||||
history=('IPython.core.historyapp.HistoryApp', | ||||
"Manage the IPython history database." | ||||
), | ||||
) | ||||
Matthias Bussonnier
|
r27221 | |||
MinRK
|
r4025 | # *do* autocreate requested profile, but don't create the config file. | ||
Matthias Bussonnier
|
r28565 | auto_create = Bool(True).tag(config=True) | ||
MinRK
|
r3963 | # configurables | ||
Min RK
|
r22340 | quick = Bool(False, | ||
MinRK
|
r3963 | help="""Start IPython quickly by skipping the loading of config files.""" | ||
Min RK
|
r22340 | ).tag(config=True) | ||
@observe('quick') | ||||
def _quick_changed(self, change): | ||||
if change['new']: | ||||
MinRK
|
r3963 | self.load_config_file = lambda *a, **kw: None | ||
Min RK
|
r22340 | display_banner = Bool(True, | ||
MinRK
|
r3963 | help="Whether to display a banner upon starting IPython." | ||
Min RK
|
r22340 | ).tag(config=True) | ||
MinRK
|
r3963 | |||
# if there is code of files to run from the cmd line, don't interact | ||||
# unless the --i flag (App.force_interact) is true. | ||||
Min RK
|
r22340 | force_interact = Bool(False, | ||
MinRK
|
r3963 | help="""If a command or file is given via the command-line, | ||
Wieland Hoffmann
|
r13929 | e.g. 'ipython foo.py', start an interactive shell after executing the | ||
file or command.""" | ||||
Min RK
|
r22340 | ).tag(config=True) | ||
@observe('force_interact') | ||||
def _force_interact_changed(self, change): | ||||
if change['new']: | ||||
MinRK
|
r3963 | self.interact = True | ||
Bernardo B. Marques
|
r4872 | |||
Min RK
|
r22340 | @observe('file_to_run', 'code_to_run', 'module_to_run') | ||
def _file_to_run_changed(self, change): | ||||
new = change['new'] | ||||
Bradley M. Froehle
|
r6081 | if new: | ||
self.something_to_run = True | ||||
MinRK
|
r3963 | if new and not self.force_interact: | ||
self.interact = False | ||||
# internal, not-configurable | ||||
Bradley M. Froehle
|
r6081 | something_to_run=Bool(False) | ||
MinRK
|
r3963 | |||
MinRK
|
r5214 | @catch_config_error | ||
MinRK
|
r3963 | def initialize(self, argv=None): | ||
"""Do actions after construct, but before starting the app.""" | ||||
MinRK
|
r3968 | super(TerminalIPythonApp, self).initialize(argv) | ||
MinRK
|
r3982 | if self.subapp is not None: | ||
# don't bother initializing further, starting subapp | ||||
return | ||||
Antony Lee
|
r28756 | # print(self.extra_args) | ||
Bradley M. Froehle
|
r6081 | if self.extra_args and not self.something_to_run: | ||
MinRK
|
r3963 | self.file_to_run = self.extra_args[0] | ||
Bradley M. Froehle
|
r6695 | self.init_path() | ||
MinRK
|
r3963 | # create the shell | ||
self.init_shell() | ||||
# and draw the banner | ||||
self.init_banner() | ||||
# Now a variety of things that happen after the banner is printed. | ||||
self.init_gui_pylab() | ||||
self.init_extensions() | ||||
self.init_code() | ||||
def init_shell(self): | ||||
"""initialize the InteractiveShell instance""" | ||||
Brian Granger
|
r2731 | # Create an InteractiveShell instance. | ||
Bernardo B. Marques
|
r4872 | # shell.display_banner should always be False for the terminal | ||
Brian Granger
|
r2252 | # based app, because we call shell.show_banner() by hand below | ||
# so the banner shows *before* all extension loading stuff. | ||||
Matthias Bussonnier
|
r23505 | self.shell = self.interactive_shell_class.instance(parent=self, | ||
Min RK
|
r22168 | profile_dir=self.profile_dir, | ||
Thomas Kluyver
|
r12159 | ipython_dir=self.ipython_dir, user_ns=self.user_ns) | ||
MinRK
|
r5315 | self.shell.configurables.append(self) | ||
Brian Granger
|
r2252 | |||
MinRK
|
r3963 | def init_banner(self): | ||
"""optionally display the banner""" | ||||
if self.display_banner and self.interact: | ||||
self.shell.show_banner() | ||||
Brian Granger
|
r2252 | # Make sure there is a space below the banner. | ||
Thomas Kluyver
|
r13348 | if self.log_level <= logging.INFO: print() | ||
Brian Granger
|
r2252 | |||
Bradley M. Froehle
|
r7096 | def _pylab_changed(self, name, old, new): | ||
"""Replace --pylab='inline' with --pylab='auto'""" | ||||
if new == 'inline': | ||||
Min RK
|
r22340 | warnings.warn("'inline' not available as pylab backend, " | ||
Thomas Kluyver
|
r8223 | "using 'auto' instead.") | ||
Bradley M. Froehle
|
r7096 | self.pylab = 'auto' | ||
Brian Granger
|
r2252 | |||
MinRK
|
r3963 | def start(self): | ||
MinRK
|
r3982 | if self.subapp is not None: | ||
return self.subapp.start() | ||||
MinRK
|
r3963 | # perform any prexec steps: | ||
if self.interact: | ||||
Brian Granger
|
r2253 | self.log.debug("Starting IPython's mainloop...") | ||
self.shell.mainloop() | ||||
Fernando Perez
|
r2391 | else: | ||
MinRK
|
r3963 | self.log.debug("IPython not interactive...") | ||
Maciej Goszczycki
|
r27831 | self.shell.restore_term_title() | ||
Thomas Kluyver
|
r23933 | if not self.shell.last_execution_succeeded: | ||
sys.exit(1) | ||||
Brian Granger
|
r2202 | |||
Brian Granger
|
r2322 | def load_default_config(ipython_dir=None): | ||
"""Load the default config file from the default ipython_dir. | ||||
Brian Granger
|
r2245 | |||
This is useful for embedded shells. | ||||
""" | ||||
Brian Granger
|
r2322 | if ipython_dir is None: | ||
ipython_dir = get_ipython_dir() | ||||
Matthias BUSSONNIER
|
r13771 | |||
MinRK
|
r3963 | profile_dir = os.path.join(ipython_dir, 'profile_default') | ||
Min RK
|
r23784 | app = TerminalIPythonApp() | ||
app.config_file_paths.append(profile_dir) | ||||
app.load_config_file() | ||||
return app.config | ||||
Brian Granger
|
r2245 | |||
MinRK
|
r11176 | launch_new_instance = TerminalIPythonApp.launch_instance | ||