##// END OF EJS Templates
Fully refactored subprocess handling on all platforms....
Fully refactored subprocess handling on all platforms. Now we have all process-related code in utils.process, which itself imports from platform-specific files as needed. On posix, we have reliable asynchronous delivery of stdout and stderr, and on win32 at least we have the basics that subprocess.py provides, since pexpect is not available. We also now support robust killing of subprocesses that may capture SIGINT: one SIGINT on our end is sent to the subprocess, but then we kill it, to prevent a rogue subprocess from hijacking the ipython console. Note that on posix, we now depend on pexpect, but we ship our own copy to users which we'll use if there's no system pexpect installed. UNC path handling for windows was implemented as a context manager called AvoidUNCPath.

File last commit:

r1234:52b55407
r2908:06dcbd43
Show More
magic.py
147 lines | 4.3 KiB | text/x-python | PythonLexer
# encoding: utf-8
__docformat__ = "restructuredtext en"
#-------------------------------------------------------------------------------
# Copyright (C) 2008 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 os
import __builtin__
# Local imports.
from util import Bunch
# fixme: RTK thinks magics should be implemented as separate classes rather than
# methods on a single class. This would give us the ability to plug new magics
# in and configure them separately.
class Magic(object):
""" An object that maintains magic functions.
"""
def __init__(self, interpreter, config=None):
# A reference to the interpreter.
self.interpreter = interpreter
# A reference to the configuration object.
if config is None:
# fixme: we need a better place to store this information.
config = Bunch(ESC_MAGIC='%')
self.config = config
def has_magic(self, name):
""" Return True if this object provides a given magic.
Parameters
----------
name : str
"""
return hasattr(self, 'magic_' + name)
def object_find(self, name):
""" Find an object in the available namespaces.
fixme: this should probably be moved elsewhere. The interpreter?
"""
name = name.strip()
# Namespaces to search.
# fixme: implement internal and alias namespaces.
user_ns = self.interpreter.user_ns
internal_ns = {}
builtin_ns = __builtin__.__dict__
alias_ns = {}
# Order the namespaces.
namespaces = [
('Interactive', user_ns),
('IPython internal', internal_ns),
('Python builtin', builtin_ns),
('Alias', alias_ns),
]
# Initialize all results.
found = False
obj = None
space = None
ds = None
ismagic = False
isalias = False
# Look for the given name by splitting it in parts. If the head is
# found, then we look for all the remaining parts as members, and only
# declare success if we can find them all.
parts = name.split('.')
head, rest = parts[0], parts[1:]
for nsname, ns in namespaces:
try:
obj = ns[head]
except KeyError:
continue
else:
for part in rest:
try:
obj = getattr(obj, part)
except:
# Blanket except b/c some badly implemented objects
# allow __getattr__ to raise exceptions other than
# AttributeError, which then crashes us.
break
else:
# If we finish the for loop (no break), we got all members
found = True
space = nsname
isalias = (ns == alias_ns)
break # namespace loop
# Try to see if it is a magic.
if not found:
if name.startswith(self.config.ESC_MAGIC):
name = name[1:]
obj = getattr(self, 'magic_' + name, None)
if obj is not None:
found = True
space = 'IPython internal'
ismagic = True
# Last try: special-case some literals like '', [], {}, etc:
if not found and head in ["''", '""', '[]', '{}', '()']:
obj = eval(head)
found = True
space = 'Interactive'
return dict(
found=found,
obj=obj,
namespace=space,
ismagic=ismagic,
isalias=isalias,
)
def magic_pwd(self, parameter_s=''):
""" Return the current working directory path.
"""
return os.getcwd()
def magic_env(self, parameter_s=''):
""" List environment variables.
"""
return os.environ.data