application.py
397 lines
| 15.4 KiB
| text/x-python
|
PythonLexer
Brian Granger
|
r2185 | # encoding: utf-8 | ||
""" | ||||
Brian Granger
|
r2301 | An application for IPython. | ||
All top-level applications should use the classes in this module for | ||||
MinRK
|
r12814 | handling configuration and creating configurables. | ||
Brian Granger
|
r2301 | |||
Bernardo B. Marques
|
r4872 | The job of an :class:`Application` is to create the master configuration | ||
Brian Granger
|
r2731 | object and then create the configurable objects, passing the config to them. | ||
Brian Granger
|
r2185 | """ | ||
Min RK
|
r18527 | # Copyright (c) IPython Development Team. | ||
# Distributed under the terms of the Modified BSD License. | ||||
Brian Granger
|
r2185 | |||
MinRK
|
r4994 | import atexit | ||
MinRK
|
r4122 | import glob | ||
Brian Granger
|
r2252 | import logging | ||
Brian Granger
|
r2200 | import os | ||
MinRK
|
r4023 | import shutil | ||
Brian Granger
|
r2185 | import sys | ||
Brian Granger
|
r2245 | |||
Min RK
|
r21253 | from traitlets.config.application import Application, catch_config_error | ||
from traitlets.config.loader import ConfigFileNotFound, PyFileConfigLoader | ||||
Fernando Perez
|
r2403 | from IPython.core import release, crashhandler | ||
MinRK
|
r4024 | from IPython.core.profiledir import ProfileDir, ProfileDirError | ||
Min RK
|
r21253 | from IPython.paths import get_ipython_dir, get_ipython_package_dir | ||
from IPython.utils.path import ensure_dir_exists | ||||
Thomas Kluyver
|
r13447 | from IPython.utils import py3compat | ||
Min RK
|
r21253 | from traitlets import List, Unicode, Type, Bool, Dict, Set, Instance, Undefined | ||
Brian Granger
|
r2185 | |||
Min RK
|
r18527 | if os.name == 'nt': | ||
programdata = os.environ.get('PROGRAMDATA', None) | ||||
if programdata: | ||||
Min RK
|
r18542 | SYSTEM_CONFIG_DIRS = [os.path.join(programdata, 'ipython')] | ||
Min RK
|
r18527 | else: # PROGRAMDATA is not defined by default on XP. | ||
SYSTEM_CONFIG_DIRS = [] | ||||
else: | ||||
SYSTEM_CONFIG_DIRS = [ | ||||
"/usr/local/etc/ipython", | ||||
"/etc/ipython", | ||||
] | ||||
Brian Granger
|
r2185 | |||
MinRK
|
r4023 | |||
# aliases and flags | ||||
MinRK
|
r4214 | base_aliases = { | ||
MinRK
|
r12282 | 'profile-dir' : 'ProfileDir.location', | ||
MinRK
|
r4214 | 'profile' : 'BaseIPythonApplication.profile', | ||
'ipython-dir' : 'BaseIPythonApplication.ipython_dir', | ||||
'log-level' : 'Application.log_level', | ||||
MinRK
|
r11277 | 'config' : 'BaseIPythonApplication.extra_config_file', | ||
MinRK
|
r4214 | } | ||
Brian Granger
|
r2294 | |||
MinRK
|
r4023 | base_flags = dict( | ||
debug = ({'Application' : {'log_level' : logging.DEBUG}}, | ||||
"set log level to logging.DEBUG (maximize logging output)"), | ||||
quiet = ({'Application' : {'log_level' : logging.CRITICAL}}, | ||||
"set log level to logging.CRITICAL (minimize logging output)"), | ||||
init = ({'BaseIPythonApplication' : { | ||||
'copy_config_files' : True, | ||||
'auto_create' : True} | ||||
MinRK
|
r4247 | }, """Initialize profile with default config files. This is equivalent | ||
to running `ipython profile create <profile>` prior to startup. | ||||
""") | ||||
MinRK
|
r4023 | ) | ||
Min RK
|
r20868 | class ProfileAwareConfigLoader(PyFileConfigLoader): | ||
"""A Python file config loader that is aware of IPython profiles.""" | ||||
def load_subconfig(self, fname, path=None, profile=None): | ||||
if profile is not None: | ||||
try: | ||||
profile_dir = ProfileDir.find_profile_dir_by_name( | ||||
get_ipython_dir(), | ||||
profile, | ||||
) | ||||
except ProfileDirError: | ||||
return | ||||
path = profile_dir.location | ||||
return super(ProfileAwareConfigLoader, self).load_subconfig(fname, path=path) | ||||
MinRK
|
r4023 | |||
class BaseIPythonApplication(Application): | ||||
name = Unicode(u'ipython') | ||||
description = Unicode(u'IPython: an enhanced interactive Python shell.') | ||||
version = Unicode(release.version) | ||||
Bernardo B. Marques
|
r4872 | |||
MinRK
|
r4023 | aliases = Dict(base_aliases) | ||
flags = Dict(base_flags) | ||||
MinRK
|
r4114 | classes = List([ProfileDir]) | ||
Min RK
|
r20868 | |||
# enable `load_subconfig('cfg.py', profile='name')` | ||||
python_config_loader_class = ProfileAwareConfigLoader | ||||
Bernardo B. Marques
|
r4872 | |||
MinRK
|
r4023 | # Track whether the config_file has changed, | ||
# because some logic happens only if we aren't using the default. | ||||
MinRK
|
r11277 | config_file_specified = Set() | ||
Bernardo B. Marques
|
r4872 | |||
Jonathan Frederic
|
r11364 | config_file_name = Unicode() | ||
MinRK
|
r4025 | def _config_file_name_default(self): | ||
return self.name.replace('-','_') + u'_config.py' | ||||
MinRK
|
r4023 | def _config_file_name_changed(self, name, old, new): | ||
if new != old: | ||||
MinRK
|
r11277 | self.config_file_specified.add(new) | ||
MinRK
|
r4023 | |||
# The directory that contains IPython's builtin profiles. | ||||
builtin_profile_dir = Unicode( | ||||
os.path.join(get_ipython_package_dir(), u'config', u'profile', u'default') | ||||
) | ||||
Min RK
|
r18527 | |||
Jason Grout
|
r21534 | config_file_paths = List(Unicode()) | ||
MinRK
|
r4023 | def _config_file_paths_default(self): | ||
Thomas Kluyver
|
r13447 | return [py3compat.getcwd()] | ||
MinRK
|
r4023 | |||
MinRK
|
r11277 | extra_config_file = Unicode(config=True, | ||
help="""Path to an extra config file to load. | ||||
If specified, load this config file in addition to any other IPython config. | ||||
""") | ||||
def _extra_config_file_changed(self, name, old, new): | ||||
try: | ||||
self.config_files.remove(old) | ||||
except ValueError: | ||||
pass | ||||
self.config_file_specified.add(new) | ||||
self.config_files.append(new) | ||||
Thomas Kluyver
|
r5259 | profile = Unicode(u'default', config=True, | ||
MinRK
|
r4023 | help="""The IPython profile to use.""" | ||
) | ||||
Bernardo B. Marques
|
r4872 | |||
MinRK
|
r4023 | def _profile_changed(self, name, old, new): | ||
self.builtin_profile_dir = os.path.join( | ||||
get_ipython_package_dir(), u'config', u'profile', new | ||||
) | ||||
Bernardo B. Marques
|
r4872 | |||
MinRK
|
r12814 | ipython_dir = Unicode(config=True, | ||
MinRK
|
r4023 | help=""" | ||
The name of the IPython directory. This directory is used for logging | ||||
configuration (through profiles), history storage, etc. The default | ||||
Thomas Kluyver
|
r16597 | is usually $HOME/.ipython. This option can also be specified through | ||
Bradley M. Froehle
|
r6696 | the environment variable IPYTHONDIR. | ||
Brian Granger
|
r2294 | """ | ||
MinRK
|
r4023 | ) | ||
MinRK
|
r12814 | def _ipython_dir_default(self): | ||
d = get_ipython_dir() | ||||
MinRK
|
r12850 | self._ipython_dir_changed('ipython_dir', d, d) | ||
MinRK
|
r12814 | return d | ||
MinRK
|
r11836 | _in_init_profile_dir = False | ||
Sylvain Corlay
|
r20946 | profile_dir = Instance(ProfileDir, allow_none=True) | ||
MinRK
|
r11836 | def _profile_dir_default(self): | ||
# avoid recursion | ||||
if self._in_init_profile_dir: | ||||
return | ||||
# profile_dir requested early, force initialization | ||||
self.init_profile_dir() | ||||
return self.profile_dir | ||||
Bernardo B. Marques
|
r4872 | |||
MinRK
|
r4023 | overwrite = Bool(False, config=True, | ||
help="""Whether to overwrite existing config files when copying""") | ||||
auto_create = Bool(False, config=True, | ||||
help="""Whether to create profile dir if it doesn't exist""") | ||||
Bernardo B. Marques
|
r4872 | |||
Jason Grout
|
r21534 | config_files = List(Unicode()) | ||
MinRK
|
r4023 | def _config_files_default(self): | ||
Jonathan Frederic
|
r11364 | return [self.config_file_name] | ||
Bernardo B. Marques
|
r4872 | |||
MinRK
|
r4023 | copy_config_files = Bool(False, config=True, | ||
MinRK
|
r4025 | help="""Whether to install the default config files into the profile dir. | ||
If a new profile is being created, and IPython contains config files for that | ||||
profile, then they will be staged into the new directory. Otherwise, | ||||
default config files will be automatically generated. | ||||
""") | ||||
MinRK
|
r5317 | |||
verbose_crash = Bool(False, config=True, | ||||
Jason Grout
|
r6872 | help="""Create a massive crash report when IPython encounters what may be an | ||
MinRK
|
r5317 | internal error. The default is to append a short message to the | ||
usual traceback""") | ||||
MinRK
|
r4023 | |||
# The class to use as the crash handler. | ||||
crash_handler_class = Type(crashhandler.CrashHandler) | ||||
Thomas Spura
|
r10381 | @catch_config_error | ||
MinRK
|
r4023 | def __init__(self, **kwargs): | ||
super(BaseIPythonApplication, self).__init__(**kwargs) | ||||
Thomas Spura
|
r10154 | # ensure current working directory exists | ||
Thomas Spura
|
r10169 | try: | ||
Thomas Kluyver
|
r13447 | directory = py3compat.getcwd() | ||
Thomas Spura
|
r10169 | except: | ||
Min RK
|
r20485 | # exit if cwd doesn't exist | ||
Thomas Spura
|
r10169 | self.log.error("Current working directory doesn't exist.") | ||
Min RK
|
r20485 | self.exit(1) | ||
Thomas Spura
|
r10154 | |||
MinRK
|
r4023 | #------------------------------------------------------------------------- | ||
# Various stages of Application creation | ||||
#------------------------------------------------------------------------- | ||||
Brian Granger
|
r2200 | |||
MinRK
|
r4023 | def init_crash_handler(self): | ||
"""Create a crash handler, typically setting sys.excepthook to it.""" | ||||
self.crash_handler = self.crash_handler_class(self) | ||||
MinRK
|
r5317 | sys.excepthook = self.excepthook | ||
MinRK
|
r4994 | def unset_crashhandler(): | ||
sys.excepthook = sys.__excepthook__ | ||||
atexit.register(unset_crashhandler) | ||||
MinRK
|
r5317 | |||
def excepthook(self, etype, evalue, tb): | ||||
"""this is sys.excepthook after init_crashhandler | ||||
set self.verbose_crash=True to use our full crashhandler, instead of | ||||
a regular traceback with a short message (crash_handler_lite) | ||||
""" | ||||
if self.verbose_crash: | ||||
return self.crash_handler(etype, evalue, tb) | ||||
else: | ||||
return crashhandler.crash_handler_lite(etype, evalue, tb) | ||||
MinRK
|
r4023 | def _ipython_dir_changed(self, name, old, new): | ||
Sylvain Corlay
|
r21172 | if old is not Undefined: | ||
Thomas Kluyver
|
r19121 | str_old = py3compat.cast_bytes_py2(os.path.abspath(old), | ||
sys.getfilesystemencoding() | ||||
) | ||||
if str_old in sys.path: | ||||
sys.path.remove(str_old) | ||||
MinRK
|
r14693 | str_path = py3compat.cast_bytes_py2(os.path.abspath(new), | ||
sys.getfilesystemencoding() | ||||
) | ||||
sys.path.append(str_path) | ||||
MinRK
|
r16486 | ensure_dir_exists(new) | ||
MinRK
|
r4023 | readme = os.path.join(new, 'README') | ||
MinRK
|
r12814 | readme_src = os.path.join(get_ipython_package_dir(), u'config', u'profile', 'README') | ||
if not os.path.exists(readme) and os.path.exists(readme_src): | ||||
shutil.copy(readme_src, readme) | ||||
for d in ('extensions', 'nbextensions'): | ||||
path = os.path.join(new, d) | ||||
MinRK
|
r16486 | try: | ||
ensure_dir_exists(path) | ||||
Matthias Bussonnier
|
r21298 | except OSError as e: | ||
MinRK
|
r16486 | # this will not be EEXIST | ||
self.log.error("couldn't create path %s: %s", path, e) | ||||
Bradley M. Froehle
|
r6697 | self.log.debug("IPYTHONDIR set to: %s" % new) | ||
MinRK
|
r4023 | |||
def load_config_file(self, suppress_errors=True): | ||||
Brian Granger
|
r2200 | """Load the config file. | ||
MinRK
|
r4023 | |||
Thomas Kluyver
|
r3458 | By default, errors in loading config are handled, and a warning | ||
printed on screen. For testing, the suppress_errors option is set | ||||
to False, so errors will make tests fail. | ||||
Brian Granger
|
r2200 | """ | ||
MinRK
|
r4564 | self.log.debug("Searching path %s for config files", self.config_file_paths) | ||
MinRK
|
r4025 | base_config = 'ipython_config.py' | ||
self.log.debug("Attempting to load config file: %s" % | ||||
base_config) | ||||
try: | ||||
Application.load_config_file( | ||||
self, | ||||
Bernardo B. Marques
|
r4872 | base_config, | ||
MinRK
|
r4025 | path=self.config_file_paths | ||
) | ||||
MinRK
|
r4909 | except ConfigFileNotFound: | ||
MinRK
|
r4025 | # ignore errors loading parent | ||
MinRK
|
r4564 | self.log.debug("Config file %s not found", base_config) | ||
MinRK
|
r4025 | pass | ||
MinRK
|
r11277 | |||
for config_file_name in self.config_files: | ||||
if not config_file_name or config_file_name == base_config: | ||||
continue | ||||
self.log.debug("Attempting to load config file: %s" % | ||||
self.config_file_name) | ||||
try: | ||||
Application.load_config_file( | ||||
self, | ||||
config_file_name, | ||||
path=self.config_file_paths | ||||
) | ||||
except ConfigFileNotFound: | ||||
# Only warn if the default config file was NOT being used. | ||||
if config_file_name in self.config_file_specified: | ||||
msg = self.log.warn | ||||
else: | ||||
msg = self.log.debug | ||||
msg("Config file not found, skipping: %s", config_file_name) | ||||
Matthias Bussonnier
|
r21262 | except Exception: | ||
MinRK
|
r11277 | # For testing purposes. | ||
if not suppress_errors: | ||||
raise | ||||
self.log.warn("Error loading config file: %s" % | ||||
self.config_file_name, exc_info=True) | ||||
Brian Granger
|
r2294 | |||
MinRK
|
r4023 | def init_profile_dir(self): | ||
"""initialize the profile dir""" | ||||
MinRK
|
r11836 | self._in_init_profile_dir = True | ||
MinRK
|
r11870 | if self.profile_dir is not None: | ||
# already ran | ||||
return | ||||
MinRK
|
r12796 | if 'ProfileDir.location' not in self.config: | ||
MinRK
|
r4023 | # location not specified, find by profile name | ||
Brian Granger
|
r2270 | try: | ||
MinRK
|
r4023 | p = ProfileDir.find_profile_dir_by_name(self.ipython_dir, self.profile, self.config) | ||
except ProfileDirError: | ||||
# not found, maybe create it (always create default profile) | ||||
Matthias BUSSONNIER
|
r8885 | if self.auto_create or self.profile == 'default': | ||
MinRK
|
r4023 | try: | ||
p = ProfileDir.create_profile_dir_by_name(self.ipython_dir, self.profile, self.config) | ||||
except ProfileDirError: | ||||
self.log.fatal("Could not create profile: %r"%self.profile) | ||||
self.exit(1) | ||||
else: | ||||
self.log.info("Created profile dir: %r"%p.location) | ||||
else: | ||||
self.log.fatal("Profile %r not found."%self.profile) | ||||
self.exit(1) | ||||
else: | ||||
Matthias Bussonnier
|
r20725 | self.log.debug("Using existing profile dir: %r"%p.location) | ||
Brian Granger
|
r2303 | else: | ||
MinRK
|
r12796 | location = self.config.ProfileDir.location | ||
MinRK
|
r4023 | # location is fully specified | ||
try: | ||||
p = ProfileDir.find_profile_dir(location, self.config) | ||||
except ProfileDirError: | ||||
# not found, maybe create it | ||||
if self.auto_create: | ||||
try: | ||||
p = ProfileDir.create_profile_dir(location, self.config) | ||||
except ProfileDirError: | ||||
self.log.fatal("Could not create profile directory: %r"%location) | ||||
self.exit(1) | ||||
else: | ||||
Matthias Bussonnier
|
r20725 | self.log.debug("Creating new profile dir: %r"%location) | ||
MinRK
|
r4023 | else: | ||
self.log.fatal("Profile directory %r not found."%location) | ||||
self.exit(1) | ||||
else: | ||||
self.log.info("Using existing profile dir: %r"%location) | ||||
MinRK
|
r14790 | # if profile_dir is specified explicitly, set profile name | ||
dir_name = os.path.basename(p.location) | ||||
if dir_name.startswith('profile_'): | ||||
self.profile = dir_name[8:] | ||||
Bernardo B. Marques
|
r4872 | |||
MinRK
|
r4023 | self.profile_dir = p | ||
self.config_file_paths.append(p.location) | ||||
MinRK
|
r11836 | self._in_init_profile_dir = False | ||
Bernardo B. Marques
|
r4872 | |||
MinRK
|
r4023 | def init_config_files(self): | ||
"""[optionally] copy default config files into profile dir.""" | ||||
Min RK
|
r18527 | self.config_file_paths.extend(SYSTEM_CONFIG_DIRS) | ||
MinRK
|
r4023 | # copy config files | ||
MinRK
|
r4122 | path = self.builtin_profile_dir | ||
MinRK
|
r4023 | if self.copy_config_files: | ||
src = self.profile | ||||
Bernardo B. Marques
|
r4872 | |||
MinRK
|
r4025 | cfg = self.config_file_name | ||
if path and os.path.exists(os.path.join(path, cfg)): | ||||
self.log.warn("Staging %r from %s into %r [overwrite=%s]"%( | ||||
cfg, src, self.profile_dir.location, self.overwrite) | ||||
) | ||||
MinRK
|
r4023 | self.profile_dir.copy_config_file(cfg, path=path, overwrite=self.overwrite) | ||
MinRK
|
r4025 | else: | ||
self.stage_default_config_file() | ||||
MinRK
|
r4122 | else: | ||
# Still stage *bundled* config files, but not generated ones | ||||
# This is necessary for `ipython profile=sympy` to load the profile | ||||
# on the first go | ||||
files = glob.glob(os.path.join(path, '*.py')) | ||||
for fullpath in files: | ||||
cfg = os.path.basename(fullpath) | ||||
if self.profile_dir.copy_config_file(cfg, path=path, overwrite=False): | ||||
# file was copied | ||||
self.log.warn("Staging bundled %s from %s into %r"%( | ||||
cfg, self.profile, self.profile_dir.location) | ||||
) | ||||
Bernardo B. Marques
|
r4872 | |||
MinRK
|
r4025 | def stage_default_config_file(self): | ||
"""auto generate default config file, and stage it into the profile.""" | ||||
s = self.generate_config_file() | ||||
fname = os.path.join(self.profile_dir.location, self.config_file_name) | ||||
if self.overwrite or not os.path.exists(fname): | ||||
self.log.warn("Generating default config file: %r"%(fname)) | ||||
with open(fname, 'w') as f: | ||||
f.write(s) | ||||
Bernardo B. Marques
|
r4872 | |||
MinRK
|
r5214 | @catch_config_error | ||
MinRK
|
r4023 | def initialize(self, argv=None): | ||
MinRK
|
r4106 | # don't hook up crash handler before parsing command-line | ||
MinRK
|
r4023 | self.parse_command_line(argv) | ||
MinRK
|
r4106 | self.init_crash_handler() | ||
MinRK
|
r4025 | if self.subapp is not None: | ||
# stop here if subapp is taking over | ||||
return | ||||
MinRK
|
r4023 | cl_config = self.config | ||
self.init_profile_dir() | ||||
self.init_config_files() | ||||
self.load_config_file() | ||||
# enforce cl-opts override configfile opts: | ||||
self.update_config(cl_config) | ||||
Brian Granger
|
r2296 | |||