|
|
# encoding: utf-8
|
|
|
"""
|
|
|
An application for IPython.
|
|
|
|
|
|
All top-level applications should use the classes in this module for
|
|
|
handling configuration and creating componenets.
|
|
|
|
|
|
The job of an :class:`Application` is to create the master configuration
|
|
|
object and then create the configurable objects, passing the config to them.
|
|
|
|
|
|
Authors:
|
|
|
|
|
|
* Brian Granger
|
|
|
* Fernando Perez
|
|
|
|
|
|
Notes
|
|
|
-----
|
|
|
"""
|
|
|
|
|
|
#-----------------------------------------------------------------------------
|
|
|
# Copyright (C) 2008-2009 The IPython Development Team
|
|
|
#
|
|
|
# Distributed under the terms of the BSD License. The full license is in
|
|
|
# the file COPYING, distributed as part of this software.
|
|
|
#-----------------------------------------------------------------------------
|
|
|
|
|
|
#-----------------------------------------------------------------------------
|
|
|
# Imports
|
|
|
#-----------------------------------------------------------------------------
|
|
|
|
|
|
import logging
|
|
|
import os
|
|
|
import sys
|
|
|
|
|
|
from IPython.core import release, crashhandler
|
|
|
from IPython.utils.path import get_ipython_dir, get_ipython_package_dir
|
|
|
from IPython.config.loader import (
|
|
|
PyFileConfigLoader,
|
|
|
ArgParseConfigLoader,
|
|
|
Config,
|
|
|
)
|
|
|
|
|
|
#-----------------------------------------------------------------------------
|
|
|
# Classes and functions
|
|
|
#-----------------------------------------------------------------------------
|
|
|
|
|
|
class ApplicationError(Exception):
|
|
|
pass
|
|
|
|
|
|
|
|
|
class BaseAppConfigLoader(ArgParseConfigLoader):
|
|
|
"""Default command line options for IPython based applications."""
|
|
|
|
|
|
def _add_ipython_dir(self, parser):
|
|
|
"""Add the --ipython-dir option to the parser."""
|
|
|
paa = parser.add_argument
|
|
|
paa('--ipython-dir',
|
|
|
dest='Global.ipython_dir',type=unicode,
|
|
|
help=
|
|
|
"""Set to override default location of the IPython directory
|
|
|
IPYTHON_DIR, stored as Global.ipython_dir. This can also be
|
|
|
specified through the environment variable IPYTHON_DIR.""",
|
|
|
metavar='Global.ipython_dir')
|
|
|
|
|
|
def _add_log_level(self, parser):
|
|
|
"""Add the --log-level option to the parser."""
|
|
|
paa = parser.add_argument
|
|
|
paa('--log-level',
|
|
|
dest="Global.log_level",type=int,
|
|
|
help='Set the log level (0,10,20,30,40,50). Default is 30.',
|
|
|
metavar='Global.log_level')
|
|
|
|
|
|
def _add_version(self, parser):
|
|
|
"""Add the --version option to the parser."""
|
|
|
parser.add_argument('--version', action="version",
|
|
|
version=self.version)
|
|
|
|
|
|
def _add_arguments(self):
|
|
|
self._add_ipython_dir(self.parser)
|
|
|
self._add_log_level(self.parser)
|
|
|
try: # Old versions of argparse don't have a version action
|
|
|
self._add_version(self.parser)
|
|
|
except Exception:
|
|
|
pass
|
|
|
|
|
|
|
|
|
class Application(object):
|
|
|
"""Load a config, construct configurables and set them running.
|
|
|
|
|
|
The configuration of an application can be done via three different Config
|
|
|
objects, which are loaded and ultimately merged into a single one used
|
|
|
from that point on by the app. These are:
|
|
|
|
|
|
1. default_config: internal defaults, implemented in code.
|
|
|
2. file_config: read from the filesystem.
|
|
|
3. command_line_config: read from the system's command line flags.
|
|
|
|
|
|
During initialization, 3 is actually read before 2, since at the
|
|
|
command-line one may override the location of the file to be read. But the
|
|
|
above is the order in which the merge is made.
|
|
|
"""
|
|
|
|
|
|
name = u'ipython'
|
|
|
description = 'IPython: an enhanced interactive Python shell.'
|
|
|
#: Usage message printed by argparse. If None, auto-generate
|
|
|
usage = None
|
|
|
#: The command line config loader. Subclass of ArgParseConfigLoader.
|
|
|
command_line_loader = BaseAppConfigLoader
|
|
|
#: The name of the config file to load, determined at runtime
|
|
|
config_file_name = None
|
|
|
#: The name of the default config file. Track separately from the actual
|
|
|
#: name because some logic happens only if we aren't using the default.
|
|
|
default_config_file_name = u'ipython_config.py'
|
|
|
default_log_level = logging.WARN
|
|
|
#: Set by --profile option
|
|
|
profile_name = None
|
|
|
#: User's ipython directory, typically ~/.ipython or ~/.config/ipython/
|
|
|
ipython_dir = None
|
|
|
#: Internal defaults, implemented in code.
|
|
|
default_config = None
|
|
|
#: Read from the filesystem.
|
|
|
file_config = None
|
|
|
#: Read from the system's command line flags.
|
|
|
command_line_config = None
|
|
|
#: The final config that will be passed to the main object.
|
|
|
master_config = None
|
|
|
#: A reference to the argv to be used (typically ends up being sys.argv[1:])
|
|
|
argv = None
|
|
|
#: extra arguments computed by the command-line loader
|
|
|
extra_args = None
|
|
|
#: The class to use as the crash handler.
|
|
|
crash_handler_class = crashhandler.CrashHandler
|
|
|
|
|
|
# Private attributes
|
|
|
_exiting = False
|
|
|
_initialized = False
|
|
|
|
|
|
def __init__(self, argv=None):
|
|
|
self.argv = sys.argv[1:] if argv is None else argv
|
|
|
self.init_logger()
|
|
|
|
|
|
def init_logger(self):
|
|
|
self.log = logging.getLogger(self.__class__.__name__)
|
|
|
# This is used as the default until the command line arguments are read.
|
|
|
self.log.setLevel(self.default_log_level)
|
|
|
self._log_handler = logging.StreamHandler()
|
|
|
self._log_formatter = logging.Formatter("[%(name)s] %(message)s")
|
|
|
self._log_handler.setFormatter(self._log_formatter)
|
|
|
self.log.addHandler(self._log_handler)
|
|
|
|
|
|
def _set_log_level(self, level):
|
|
|
self.log.setLevel(level)
|
|
|
|
|
|
def _get_log_level(self):
|
|
|
return self.log.level
|
|
|
|
|
|
log_level = property(_get_log_level, _set_log_level)
|
|
|
|
|
|
def initialize(self):
|
|
|
"""Initialize the application.
|
|
|
|
|
|
Loads all configuration information and sets all application state, but
|
|
|
does not start any relevant processing (typically some kind of event
|
|
|
loop).
|
|
|
|
|
|
Once this method has been called, the application is flagged as
|
|
|
initialized and the method becomes a no-op."""
|
|
|
|
|
|
if self._initialized:
|
|
|
return
|
|
|
|
|
|
# The first part is protected with an 'attempt' wrapper, that will log
|
|
|
# failures with the basic system traceback machinery. Once our crash
|
|
|
# handler is in place, we can let any subsequent exception propagate,
|
|
|
# as our handler will log it with much better detail than the default.
|
|
|
self.attempt(self.create_crash_handler)
|
|
|
|
|
|
# Configuration phase
|
|
|
# Default config (internally hardwired in application code)
|
|
|
self.create_default_config()
|
|
|
self.log_default_config()
|
|
|
self.set_default_config_log_level()
|
|
|
|
|
|
# Command-line config
|
|
|
self.pre_load_command_line_config()
|
|
|
self.load_command_line_config()
|
|
|
self.set_command_line_config_log_level()
|
|
|
self.post_load_command_line_config()
|
|
|
self.log_command_line_config()
|
|
|
|
|
|
# Find resources needed for filesystem access, using information from
|
|
|
# the above two
|
|
|
self.find_ipython_dir()
|
|
|
self.find_resources()
|
|
|
self.find_config_file_name()
|
|
|
self.find_config_file_paths()
|
|
|
|
|
|
# File-based config
|
|
|
self.pre_load_file_config()
|
|
|
self.load_file_config()
|
|
|
self.set_file_config_log_level()
|
|
|
self.post_load_file_config()
|
|
|
self.log_file_config()
|
|
|
|
|
|
# Merge all config objects into a single one the app can then use
|
|
|
self.merge_configs()
|
|
|
self.log_master_config()
|
|
|
|
|
|
# Construction phase
|
|
|
self.pre_construct()
|
|
|
self.construct()
|
|
|
self.post_construct()
|
|
|
|
|
|
# Done, flag as such and
|
|
|
self._initialized = True
|
|
|
|
|
|
def start(self):
|
|
|
"""Start the application."""
|
|
|
self.initialize()
|
|
|
self.start_app()
|
|
|
|
|
|
#-------------------------------------------------------------------------
|
|
|
# Various stages of Application creation
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
|
|
def create_crash_handler(self):
|
|
|
"""Create a crash handler, typically setting sys.excepthook to it."""
|
|
|
self.crash_handler = self.crash_handler_class(self)
|
|
|
sys.excepthook = self.crash_handler
|
|
|
|
|
|
def create_default_config(self):
|
|
|
"""Create defaults that can't be set elsewhere.
|
|
|
|
|
|
For the most part, we try to set default in the class attributes
|
|
|
of Configurables. But, defaults the top-level Application (which is
|
|
|
not a HasTraits or Configurables) are not set in this way. Instead
|
|
|
we set them here. The Global section is for variables like this that
|
|
|
don't belong to a particular configurable.
|
|
|
"""
|
|
|
c = Config()
|
|
|
c.Global.ipython_dir = get_ipython_dir()
|
|
|
c.Global.log_level = self.log_level
|
|
|
self.default_config = c
|
|
|
|
|
|
def log_default_config(self):
|
|
|
self.log.debug('Default config loaded:')
|
|
|
self.log.debug(repr(self.default_config))
|
|
|
|
|
|
def set_default_config_log_level(self):
|
|
|
try:
|
|
|
self.log_level = self.default_config.Global.log_level
|
|
|
except AttributeError:
|
|
|
# Fallback to the default_log_level class attribute
|
|
|
pass
|
|
|
|
|
|
def create_command_line_config(self):
|
|
|
"""Create and return a command line config loader."""
|
|
|
return self.command_line_loader(
|
|
|
self.argv,
|
|
|
description=self.description,
|
|
|
version=release.version,
|
|
|
usage=self.usage
|
|
|
)
|
|
|
|
|
|
def pre_load_command_line_config(self):
|
|
|
"""Do actions just before loading the command line config."""
|
|
|
pass
|
|
|
|
|
|
def load_command_line_config(self):
|
|
|
"""Load the command line config."""
|
|
|
loader = self.create_command_line_config()
|
|
|
self.command_line_config = loader.load_config()
|
|
|
self.extra_args = loader.get_extra_args()
|
|
|
|
|
|
def set_command_line_config_log_level(self):
|
|
|
try:
|
|
|
self.log_level = self.command_line_config.Global.log_level
|
|
|
except AttributeError:
|
|
|
pass
|
|
|
|
|
|
def post_load_command_line_config(self):
|
|
|
"""Do actions just after loading the command line config."""
|
|
|
pass
|
|
|
|
|
|
def log_command_line_config(self):
|
|
|
self.log.debug("Command line config loaded:")
|
|
|
self.log.debug(repr(self.command_line_config))
|
|
|
|
|
|
def find_ipython_dir(self):
|
|
|
"""Set the IPython directory.
|
|
|
|
|
|
This sets ``self.ipython_dir``, but the actual value that is passed to
|
|
|
the application is kept in either ``self.default_config`` or
|
|
|
``self.command_line_config``. This also adds ``self.ipython_dir`` to
|
|
|
``sys.path`` so config files there can be referenced by other config
|
|
|
files.
|
|
|
"""
|
|
|
|
|
|
try:
|
|
|
self.ipython_dir = self.command_line_config.Global.ipython_dir
|
|
|
except AttributeError:
|
|
|
self.ipython_dir = self.default_config.Global.ipython_dir
|
|
|
sys.path.append(os.path.abspath(self.ipython_dir))
|
|
|
if not os.path.isdir(self.ipython_dir):
|
|
|
os.makedirs(self.ipython_dir, mode=0777)
|
|
|
self.log.debug("IPYTHON_DIR set to: %s" % self.ipython_dir)
|
|
|
|
|
|
def find_resources(self):
|
|
|
"""Find other resources that need to be in place.
|
|
|
|
|
|
Things like cluster directories need to be in place to find the
|
|
|
config file. These happen right after the IPython directory has
|
|
|
been set.
|
|
|
"""
|
|
|
pass
|
|
|
|
|
|
def find_config_file_name(self):
|
|
|
"""Find the config file name for this application.
|
|
|
|
|
|
This must set ``self.config_file_name`` to the filename of the
|
|
|
config file to use (just the filename). The search paths for the
|
|
|
config file are set in :meth:`find_config_file_paths` and then passed
|
|
|
to the config file loader where they are resolved to an absolute path.
|
|
|
|
|
|
If a profile has been set at the command line, this will resolve it.
|
|
|
"""
|
|
|
try:
|
|
|
self.config_file_name = self.command_line_config.Global.config_file
|
|
|
except AttributeError:
|
|
|
pass
|
|
|
else:
|
|
|
return
|
|
|
|
|
|
try:
|
|
|
self.profile_name = self.command_line_config.Global.profile
|
|
|
except AttributeError:
|
|
|
# Just use the default as there is no profile
|
|
|
self.config_file_name = self.default_config_file_name
|
|
|
else:
|
|
|
# Use the default config file name and profile name if set
|
|
|
# to determine the used config file name.
|
|
|
name_parts = self.default_config_file_name.split('.')
|
|
|
name_parts.insert(1, u'_' + self.profile_name + u'.')
|
|
|
self.config_file_name = ''.join(name_parts)
|
|
|
|
|
|
def find_config_file_paths(self):
|
|
|
"""Set the search paths for resolving the config file.
|
|
|
|
|
|
This must set ``self.config_file_paths`` to a sequence of search
|
|
|
paths to pass to the config file loader.
|
|
|
"""
|
|
|
# Include our own profiles directory last, so that users can still find
|
|
|
# our shipped copies of builtin profiles even if they don't have them
|
|
|
# in their local ipython directory.
|
|
|
prof_dir = os.path.join(get_ipython_package_dir(), 'config', 'profile')
|
|
|
self.config_file_paths = (os.getcwdu(), self.ipython_dir, prof_dir)
|
|
|
|
|
|
def pre_load_file_config(self):
|
|
|
"""Do actions before the config file is loaded."""
|
|
|
pass
|
|
|
|
|
|
def load_file_config(self, suppress_errors=True):
|
|
|
"""Load the config file.
|
|
|
|
|
|
This tries to load the config file from disk. If successful, the
|
|
|
``CONFIG_FILE`` config variable is set to the resolved config file
|
|
|
location. If not successful, an empty config is used.
|
|
|
|
|
|
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.
|
|
|
"""
|
|
|
self.log.debug("Attempting to load config file: %s" %
|
|
|
self.config_file_name)
|
|
|
loader = PyFileConfigLoader(self.config_file_name,
|
|
|
path=self.config_file_paths)
|
|
|
try:
|
|
|
self.file_config = loader.load_config()
|
|
|
self.file_config.Global.config_file = loader.full_filename
|
|
|
except IOError:
|
|
|
# Only warn if the default config file was NOT being used.
|
|
|
if not self.config_file_name==self.default_config_file_name:
|
|
|
self.log.warn("Config file not found, skipping: %s" %
|
|
|
self.config_file_name, exc_info=True)
|
|
|
self.file_config = Config()
|
|
|
except:
|
|
|
if not suppress_errors: # For testing purposes
|
|
|
raise
|
|
|
self.log.warn("Error loading config file: %s" %
|
|
|
self.config_file_name, exc_info=True)
|
|
|
self.file_config = Config()
|
|
|
|
|
|
def set_file_config_log_level(self):
|
|
|
# We need to keeep self.log_level updated. But we only use the value
|
|
|
# of the file_config if a value was not specified at the command
|
|
|
# line, because the command line overrides everything.
|
|
|
if not hasattr(self.command_line_config.Global, 'log_level'):
|
|
|
try:
|
|
|
self.log_level = self.file_config.Global.log_level
|
|
|
except AttributeError:
|
|
|
pass # Use existing value
|
|
|
|
|
|
def post_load_file_config(self):
|
|
|
"""Do actions after the config file is loaded."""
|
|
|
pass
|
|
|
|
|
|
def log_file_config(self):
|
|
|
if hasattr(self.file_config.Global, 'config_file'):
|
|
|
self.log.debug("Config file loaded: %s" %
|
|
|
self.file_config.Global.config_file)
|
|
|
self.log.debug(repr(self.file_config))
|
|
|
|
|
|
def merge_configs(self):
|
|
|
"""Merge the default, command line and file config objects."""
|
|
|
config = Config()
|
|
|
config._merge(self.default_config)
|
|
|
config._merge(self.file_config)
|
|
|
config._merge(self.command_line_config)
|
|
|
|
|
|
# XXX fperez - propose to Brian we rename master_config to simply
|
|
|
# config, I think this is going to be heavily used in examples and
|
|
|
# application code and the name is shorter/easier to find/remember.
|
|
|
# For now, just alias it...
|
|
|
self.master_config = config
|
|
|
self.config = config
|
|
|
|
|
|
def log_master_config(self):
|
|
|
self.log.debug("Master config created:")
|
|
|
self.log.debug(repr(self.master_config))
|
|
|
|
|
|
def pre_construct(self):
|
|
|
"""Do actions after the config has been built, but before construct."""
|
|
|
pass
|
|
|
|
|
|
def construct(self):
|
|
|
"""Construct the main objects that make up this app."""
|
|
|
self.log.debug("Constructing main objects for application")
|
|
|
|
|
|
def post_construct(self):
|
|
|
"""Do actions after construct, but before starting the app."""
|
|
|
pass
|
|
|
|
|
|
def start_app(self):
|
|
|
"""Actually start the app."""
|
|
|
self.log.debug("Starting application")
|
|
|
|
|
|
#-------------------------------------------------------------------------
|
|
|
# Utility methods
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
|
|
def exit(self, exit_status=0):
|
|
|
if self._exiting:
|
|
|
pass
|
|
|
else:
|
|
|
self.log.debug("Exiting application: %s" % self.name)
|
|
|
self._exiting = True
|
|
|
sys.exit(exit_status)
|
|
|
|
|
|
def attempt(self, func):
|
|
|
try:
|
|
|
func()
|
|
|
except SystemExit:
|
|
|
raise
|
|
|
except:
|
|
|
self.log.critical("Aborting application: %s" % self.name,
|
|
|
exc_info=True)
|
|
|
self.exit(0)
|
|
|
|
|
|
|