##// END OF EJS Templates
More code review changes:...
Brian Granger -
Show More
@@ -0,0 +1,7
1 .. _extensions_cythonmagic:
2
3 ===========
4 cythonmagic
5 ===========
6
7 .. automodule:: IPython.extensions.cythonmagic
@@ -1,180 +1,181
1 # -*- coding: utf-8 -*-
1 # -*- coding: utf-8 -*-
2 """
2 """
3 Cython related magics.
3 Cython related magics.
4
4
5 Author:
5 Author:
6 * Brian Granger
6 * Brian Granger
7
7
8 Parts of this code were taken from Cython.inline.
8 Parts of this code were taken from Cython.inline.
9 """
9 """
10 #-----------------------------------------------------------------------------
10 #-----------------------------------------------------------------------------
11 # Copyright (C) 2010-2011, IPython Development Team.
11 # Copyright (C) 2010-2011, IPython Development Team.
12 #
12 #
13 # Distributed under the terms of the Modified BSD License.
13 # Distributed under the terms of the Modified BSD License.
14 #
14 #
15 # The full license is in the file COPYING.txt, distributed with this software.
15 # The full license is in the file COPYING.txt, distributed with this software.
16 #-----------------------------------------------------------------------------
16 #-----------------------------------------------------------------------------
17
17
18 import io
18 import os, sys
19 import os, sys
19 from importlib import import_module
20 from importlib import import_module
20 import imp
21 import imp
21
22
22 try:
23 try:
23 import hashlib
24 import hashlib
24 except ImportError:
25 except ImportError:
25 import md5 as hashlib
26 import md5 as hashlib
26
27
27 from distutils.core import Distribution, Extension
28 from distutils.core import Distribution, Extension
28 from distutils.command.build_ext import build_ext
29 from distutils.command.build_ext import build_ext
29
30
30 from IPython.core.magic import Magics, magics_class, cell_magic
31 from IPython.core.magic import Magics, magics_class, cell_magic
31 from IPython.testing.skipdoctest import skip_doctest
32 from IPython.testing.skipdoctest import skip_doctest
32 from IPython.core.magic_arguments import (
33 from IPython.core.magic_arguments import (
33 argument, magic_arguments, parse_argstring
34 argument, magic_arguments, parse_argstring
34 )
35 )
36 from IPython.utils import py3compat
35
37
36 import Cython
38 import Cython
37 from Cython.Compiler.Errors import CompileError
39 from Cython.Compiler.Errors import CompileError
38 from Cython.Compiler.Main import Context, default_options
40 from Cython.Compiler.Main import Context, default_options
39 from Cython.Build.Dependencies import cythonize
41 from Cython.Build.Dependencies import cythonize
40
42
41
43
42 @magics_class
44 @magics_class
43 class CythonMagics(Magics):
45 class CythonMagics(Magics):
44
46
45 def __init__(self, shell):
47 def __init__(self, shell):
46 super(CythonMagics,self).__init__(shell)
48 super(CythonMagics,self).__init__(shell)
47 self._reloads = {}
49 self._reloads = {}
48 self._code_cache = {}
50 self._code_cache = {}
49
51
50 def _import_all(self, module):
52 def _import_all(self, module):
51 for k,v in module.__dict__.items():
53 for k,v in module.__dict__.items():
52 if not k.startswith('__'):
54 if not k.startswith('__'):
53 self.shell.push({k:v})
55 self.shell.push({k:v})
54
56
55 @cell_magic
57 @cell_magic
56 def cython_inline(self, line, cell):
58 def cython_inline(self, line, cell):
57 """Compile and run a Cython code cell using Cython.inline.
59 """Compile and run a Cython code cell using Cython.inline.
58
60
59 This magic simply passes the body of the cell to Cython.inline
61 This magic simply passes the body of the cell to Cython.inline
60 and returns the result. If the variables `a` and `b` are defined
62 and returns the result. If the variables `a` and `b` are defined
61 in the user's namespace, here is a simple example that returns
63 in the user's namespace, here is a simple example that returns
62 their sum::
64 their sum::
63
65
64 %%cython_inline
66 %%cython_inline
65 return a+b
67 return a+b
66
68
67 For most purposes, we recommend the usage of the `%%cython` magic.
69 For most purposes, we recommend the usage of the `%%cython` magic.
68 """
70 """
69 locs = self.shell.user_global_ns
71 locs = self.shell.user_global_ns
70 globs = self.shell.user_ns
72 globs = self.shell.user_ns
71 return Cython.inline(cell, locals=locs, globals=globs)
73 return Cython.inline(cell, locals=locs, globals=globs)
72
74
73 @cell_magic
75 @cell_magic
74 def cython_pyximport(self, line, cell):
76 def cython_pyximport(self, line, cell):
75 """Compile and import a Cython code cell using pyximport.
77 """Compile and import a Cython code cell using pyximport.
76
78
77 The contents of the cell are written to a `.pyx` file in the current
79 The contents of the cell are written to a `.pyx` file in the current
78 working directory, which is then imported using `pyximport`. This
80 working directory, which is then imported using `pyximport`. This
79 magic requires a module name to be passed::
81 magic requires a module name to be passed::
80
82
81 %%cython_pyximport modulename
83 %%cython_pyximport modulename
82 def f(x):
84 def f(x):
83 return 2.0*x
85 return 2.0*x
84
86
85 The compiled module is then imported and all of its symbols are injected into
87 The compiled module is then imported and all of its symbols are injected into
86 the user's namespace. For most purposes, we recommend the usage of the
88 the user's namespace. For most purposes, we recommend the usage of the
87 `%%cython` magic.
89 `%%cython` magic.
88 """
90 """
89 module_name = line.strip()
91 module_name = line.strip()
90 if not module_name:
92 if not module_name:
91 raise ValueError('module name must be given')
93 raise ValueError('module name must be given')
92 fname = module_name + '.pyx'
94 fname = module_name + '.pyx'
93 with io.open(fname, 'w', encoding='utf-8'):
95 with io.open(fname, 'w', encoding='utf-8') as f:
94 f.write(cell)
96 f.write(cell)
95 if 'pyximport' not in sys.modules:
97 if 'pyximport' not in sys.modules:
96 import pyximport
98 import pyximport
97 pyximport.install(reload_support=True)
99 pyximport.install(reload_support=True)
98 if module_name in self._reloads:
100 if module_name in self._reloads:
99 module = self._reloads[module_name]
101 module = self._reloads[module_name]
100 reload(module)
102 reload(module)
101 else:
103 else:
102 module = import_module(module_name)
104 module = import_module(module_name)
103 self._reloads[module_name] = module
105 self._reloads[module_name] = module
104 self._import_all(module)
106 self._import_all(module)
105
107
106 @magic_arguments()
108 @magic_arguments()
107 @argument(
109 @argument(
108 '-f', '--force', action='store_true', default=False,
110 '-f', '--force', action='store_true', default=False,
109 help="Force the compilation of the pyx module even if it hasn't changed"
111 help="Force the compilation of the pyx module even if it hasn't changed"
110 )
112 )
111 @cell_magic
113 @cell_magic
112 def cython(self, line, cell):
114 def cython(self, line, cell):
113 """Compile and import everything from a Cython code cell.
115 """Compile and import everything from a Cython code cell.
114
116
115 The contents of the cell are written to a `.pyx` file in the
117 The contents of the cell are written to a `.pyx` file in the
116 directory `IPYTHONDIR/cython` using a filename with the hash of the code.
118 directory `IPYTHONDIR/cython` using a filename with the hash of the code.
117 This file is then cythonized and compiled. The resulting module
119 This file is then cythonized and compiled. The resulting module
118 is imported and all of its symbols are injected into the user's
120 is imported and all of its symbols are injected into the user's
119 namespace. The usage is similar to that of `%%cython_pyximport` but
121 namespace. The usage is similar to that of `%%cython_pyximport` but
120 you don't have to pass a module name::
122 you don't have to pass a module name::
121
123
122 %%cython
124 %%cython
123 def f(x):
125 def f(x):
124 return 2.0*x
126 return 2.0*x
125 """
127 """
126 args = parse_argstring(self.cython, line)
128 args = parse_argstring(self.cython, line)
127 code = cell if cell.endswith('\n') else cell+'\n'
129 code = cell if cell.endswith('\n') else cell+'\n'
128 # distutils.Extension cannot handle sources that a unicode
130 lib_dir=os.path.join(self.shell.ipython_dir, 'cython')
129 lib_dir=str(os.path.join(self.shell.ipython_dir,'cython'))
130 cython_include_dirs=['.']
131 cython_include_dirs=['.']
131 force=args.force
132 force=args.force
132 quiet=True
133 quiet=True
133 ctx = Context(cython_include_dirs, default_options)
134 ctx = Context(cython_include_dirs, default_options)
134 key = code, sys.version_info, sys.executable, Cython.__version__
135 key = code, sys.version_info, sys.executable, Cython.__version__
135 module_name = "_cython_magic_" + hashlib.md5(str(key).encode('utf-8')).hexdigest()
136 module_name = "_cython_magic_" + hashlib.md5(str(key).encode('utf-8')).hexdigest()
136 so_ext = [ ext for ext,_,mod_type in imp.get_suffixes() if mod_type == imp.C_EXTENSION ][0]
137 so_ext = [ ext for ext,_,mod_type in imp.get_suffixes() if mod_type == imp.C_EXTENSION ][0]
137 module_path = os.path.join(lib_dir, module_name+so_ext)
138 module_path = os.path.join(lib_dir, module_name+so_ext)
138
139
139 if not os.path.exists(lib_dir):
140 if not os.path.exists(lib_dir):
140 os.makedirs(lib_dir)
141 os.makedirs(lib_dir)
141
142
142 if force or not os.path.isfile(module_path):
143 if force or not os.path.isfile(module_path):
143 cflags = []
144 cflags = []
144 c_include_dirs = []
145 c_include_dirs = []
145 if 'numpy' in code:
146 if 'numpy' in code:
146 import numpy
147 import numpy
147 c_include_dirs.append(numpy.get_include())
148 c_include_dirs.append(numpy.get_include())
148 pyx_file = os.path.join(lib_dir, module_name + '.pyx')
149 pyx_file = os.path.join(lib_dir, module_name + '.pyx')
149 with io.open(pyx_file, 'w', encoding='utf-8'):
150 pyx_file = py3compat.unicode_to_str(pyx_file, encoding=sys.getfilesystemencoding())
151 with io.open(pyx_file, 'w', encoding='utf-8') as f:
150 f.write(code)
152 f.write(code)
151 print [pyx_file]
152 extension = Extension(
153 extension = Extension(
153 name = module_name,
154 name = module_name,
154 sources = [pyx_file],
155 sources = [pyx_file],
155 include_dirs = c_include_dirs,
156 include_dirs = c_include_dirs,
156 extra_compile_args = cflags
157 extra_compile_args = cflags
157 )
158 )
158 build_extension = build_ext(Distribution())
159 build_extension = build_ext(Distribution())
159 build_extension.finalize_options()
160 build_extension.finalize_options()
160 try:
161 try:
161 build_extension.extensions = cythonize([extension], ctx=ctx, quiet=quiet)
162 build_extension.extensions = cythonize([extension], ctx=ctx, quiet=quiet)
162 except CompileError:
163 except CompileError:
163 return
164 return
164 build_extension.build_temp = os.path.dirname(pyx_file)
165 build_extension.build_temp = os.path.dirname(pyx_file)
165 build_extension.build_lib = lib_dir
166 build_extension.build_lib = lib_dir
166 build_extension.run()
167 build_extension.run()
167 self._code_cache[key] = module_name
168 self._code_cache[key] = module_name
168
169
169 module = imp.load_dynamic(module_name, module_path)
170 module = imp.load_dynamic(module_name, module_path)
170 self._import_all(module)
171 self._import_all(module)
171
172
172
173
173 _loaded = False
174 _loaded = False
174
175
175 def load_ipython_extension(ip):
176 def load_ipython_extension(ip):
176 """Load the extension in IPython."""
177 """Load the extension in IPython."""
177 global _loaded
178 global _loaded
178 if not _loaded:
179 if not _loaded:
179 ip.register_magics(CythonMagics)
180 ip.register_magics(CythonMagics)
180 _loaded = True
181 _loaded = True
@@ -1,45 +1,47
1 # -*- coding: utf-8 -*-
1 # -*- coding: utf-8 -*-
2 """Tests for the Cython magics extension."""
2 """Tests for the Cython magics extension."""
3
3
4 import os
4 import os
5 import nose.tools as nt
5 import nose.tools as nt
6
6
7 from IPython.utils import py3compat
7
8
8 code = """def f(x):
9 code = py3compat.str_to_unicode("""def f(x):
9 return 2*x
10 return 2*x
10 """
11 """)
11
12
12 try:
13 try:
13 import Cython
14 import Cython
14 except:
15 except:
15 __test__ = False
16 __test__ = False
16
17
17 def setup():
18 def setup():
18 ip = get_ipython()
19 ip = get_ipython()
19 ip.extension_manager.load_extension('cythonmagic')
20 ip.extension_manager.load_extension('cythonmagic')
20
21
21 def test_cython_inline():
22 def test_cython_inline():
22 ip = get_ipython()
23 ip = get_ipython()
23 ip.ex('a=10; b=20')
24 ip.ex('a=10; b=20')
24 result = ip.run_cell_magic('cython_inline','','return a+b')
25 result = ip.run_cell_magic('cython_inline','','return a+b')
25 nt.assert_equals(result, 30)
26 nt.assert_equals(result, 30)
26
27
27 def test_cython_pyximport():
28 def test_cython_pyximport():
28 module_name = '_test_cython_pyximport'
29 module_name = '_test_cython_pyximport'
29 ip = get_ipython()
30 ip = get_ipython()
30 ip.run_cell_magic('cython_pyximport', module_name, code)
31 ip.run_cell_magic('cython_pyximport', module_name, code)
31 ip.ex('g = f(10)')
32 ip.ex('g = f(10)')
32 nt.assert_equals(ip.user_ns['g'], 20.0)
33 nt.assert_equals(ip.user_ns['g'], 20.0)
33 try:
34 try:
34 os.remove(module_name+'.pyx')
35 os.remove(module_name+'.pyx')
35 except OSError:
36 except OSError:
36 pass
37 pass
37
38
38 def test_cython():
39 def test_cython():
39 ip = get_ipython()
40 ip = get_ipython()
40 ip.run_cell_magic('cython', '', code)
41 ip.run_cell_magic('cython', '', code)
41 ip.ex('g = f(10)')
42 ip.ex('g = f(10)')
42 nt.assert_equals(ip.user_ns['g'], 20.0)
43 nt.assert_equals(ip.user_ns['g'], 20.0)
44
43
45
44
46
45
47
@@ -1,524 +1,528
1 # -*- coding: utf-8 -*-
1 # -*- coding: utf-8 -*-
2 """IPython Test Suite Runner.
2 """IPython Test Suite Runner.
3
3
4 This module provides a main entry point to a user script to test IPython
4 This module provides a main entry point to a user script to test IPython
5 itself from the command line. There are two ways of running this script:
5 itself from the command line. There are two ways of running this script:
6
6
7 1. With the syntax `iptest all`. This runs our entire test suite by
7 1. With the syntax `iptest all`. This runs our entire test suite by
8 calling this script (with different arguments) recursively. This
8 calling this script (with different arguments) recursively. This
9 causes modules and package to be tested in different processes, using nose
9 causes modules and package to be tested in different processes, using nose
10 or trial where appropriate.
10 or trial where appropriate.
11 2. With the regular nose syntax, like `iptest -vvs IPython`. In this form
11 2. With the regular nose syntax, like `iptest -vvs IPython`. In this form
12 the script simply calls nose, but with special command line flags and
12 the script simply calls nose, but with special command line flags and
13 plugins loaded.
13 plugins loaded.
14
14
15 """
15 """
16
16
17 #-----------------------------------------------------------------------------
17 #-----------------------------------------------------------------------------
18 # Copyright (C) 2009-2011 The IPython Development Team
18 # Copyright (C) 2009-2011 The IPython Development Team
19 #
19 #
20 # Distributed under the terms of the BSD License. The full license is in
20 # Distributed under the terms of the BSD License. The full license is in
21 # the file COPYING, distributed as part of this software.
21 # the file COPYING, distributed as part of this software.
22 #-----------------------------------------------------------------------------
22 #-----------------------------------------------------------------------------
23
23
24 #-----------------------------------------------------------------------------
24 #-----------------------------------------------------------------------------
25 # Imports
25 # Imports
26 #-----------------------------------------------------------------------------
26 #-----------------------------------------------------------------------------
27
27
28 # Stdlib
28 # Stdlib
29 import os
29 import os
30 import os.path as path
30 import os.path as path
31 import signal
31 import signal
32 import sys
32 import sys
33 import subprocess
33 import subprocess
34 import tempfile
34 import tempfile
35 import time
35 import time
36 import warnings
36 import warnings
37
37
38 # Note: monkeypatch!
38 # Note: monkeypatch!
39 # We need to monkeypatch a small problem in nose itself first, before importing
39 # We need to monkeypatch a small problem in nose itself first, before importing
40 # it for actual use. This should get into nose upstream, but its release cycle
40 # it for actual use. This should get into nose upstream, but its release cycle
41 # is slow and we need it for our parametric tests to work correctly.
41 # is slow and we need it for our parametric tests to work correctly.
42 from IPython.testing import nosepatch
42 from IPython.testing import nosepatch
43 # Now, proceed to import nose itself
43 # Now, proceed to import nose itself
44 import nose.plugins.builtin
44 import nose.plugins.builtin
45 from nose.plugins.xunit import Xunit
45 from nose.plugins.xunit import Xunit
46 from nose import SkipTest
46 from nose import SkipTest
47 from nose.core import TestProgram
47 from nose.core import TestProgram
48
48
49 # Our own imports
49 # Our own imports
50 from IPython.utils.importstring import import_item
50 from IPython.utils.importstring import import_item
51 from IPython.utils.path import get_ipython_module_path
51 from IPython.utils.path import get_ipython_module_path
52 from IPython.utils.process import find_cmd, pycmd2argv
52 from IPython.utils.process import find_cmd, pycmd2argv
53 from IPython.utils.sysinfo import sys_info
53 from IPython.utils.sysinfo import sys_info
54
54
55 from IPython.testing import globalipapp
55 from IPython.testing import globalipapp
56 from IPython.testing.plugin.ipdoctest import IPythonDoctest
56 from IPython.testing.plugin.ipdoctest import IPythonDoctest
57 from IPython.external.decorators import KnownFailure, knownfailureif
57 from IPython.external.decorators import KnownFailure, knownfailureif
58
58
59 pjoin = path.join
59 pjoin = path.join
60
60
61
61
62 #-----------------------------------------------------------------------------
62 #-----------------------------------------------------------------------------
63 # Globals
63 # Globals
64 #-----------------------------------------------------------------------------
64 #-----------------------------------------------------------------------------
65
65
66
66
67 #-----------------------------------------------------------------------------
67 #-----------------------------------------------------------------------------
68 # Warnings control
68 # Warnings control
69 #-----------------------------------------------------------------------------
69 #-----------------------------------------------------------------------------
70
70
71 # Twisted generates annoying warnings with Python 2.6, as will do other code
71 # Twisted generates annoying warnings with Python 2.6, as will do other code
72 # that imports 'sets' as of today
72 # that imports 'sets' as of today
73 warnings.filterwarnings('ignore', 'the sets module is deprecated',
73 warnings.filterwarnings('ignore', 'the sets module is deprecated',
74 DeprecationWarning )
74 DeprecationWarning )
75
75
76 # This one also comes from Twisted
76 # This one also comes from Twisted
77 warnings.filterwarnings('ignore', 'the sha module is deprecated',
77 warnings.filterwarnings('ignore', 'the sha module is deprecated',
78 DeprecationWarning)
78 DeprecationWarning)
79
79
80 # Wx on Fedora11 spits these out
80 # Wx on Fedora11 spits these out
81 warnings.filterwarnings('ignore', 'wxPython/wxWidgets release number mismatch',
81 warnings.filterwarnings('ignore', 'wxPython/wxWidgets release number mismatch',
82 UserWarning)
82 UserWarning)
83
83
84 # ------------------------------------------------------------------------------
84 # ------------------------------------------------------------------------------
85 # Monkeypatch Xunit to count known failures as skipped.
85 # Monkeypatch Xunit to count known failures as skipped.
86 # ------------------------------------------------------------------------------
86 # ------------------------------------------------------------------------------
87 def monkeypatch_xunit():
87 def monkeypatch_xunit():
88 try:
88 try:
89 knownfailureif(True)(lambda: None)()
89 knownfailureif(True)(lambda: None)()
90 except Exception as e:
90 except Exception as e:
91 KnownFailureTest = type(e)
91 KnownFailureTest = type(e)
92
92
93 def addError(self, test, err, capt=None):
93 def addError(self, test, err, capt=None):
94 if issubclass(err[0], KnownFailureTest):
94 if issubclass(err[0], KnownFailureTest):
95 err = (SkipTest,) + err[1:]
95 err = (SkipTest,) + err[1:]
96 return self.orig_addError(test, err, capt)
96 return self.orig_addError(test, err, capt)
97
97
98 Xunit.orig_addError = Xunit.addError
98 Xunit.orig_addError = Xunit.addError
99 Xunit.addError = addError
99 Xunit.addError = addError
100
100
101 #-----------------------------------------------------------------------------
101 #-----------------------------------------------------------------------------
102 # Logic for skipping doctests
102 # Logic for skipping doctests
103 #-----------------------------------------------------------------------------
103 #-----------------------------------------------------------------------------
104 def extract_version(mod):
104 def extract_version(mod):
105 return mod.__version__
105 return mod.__version__
106
106
107 def test_for(item, min_version=None, callback=extract_version):
107 def test_for(item, min_version=None, callback=extract_version):
108 """Test to see if item is importable, and optionally check against a minimum
108 """Test to see if item is importable, and optionally check against a minimum
109 version.
109 version.
110
110
111 If min_version is given, the default behavior is to check against the
111 If min_version is given, the default behavior is to check against the
112 `__version__` attribute of the item, but specifying `callback` allows you to
112 `__version__` attribute of the item, but specifying `callback` allows you to
113 extract the value you are interested in. e.g::
113 extract the value you are interested in. e.g::
114
114
115 In [1]: import sys
115 In [1]: import sys
116
116
117 In [2]: from IPython.testing.iptest import test_for
117 In [2]: from IPython.testing.iptest import test_for
118
118
119 In [3]: test_for('sys', (2,6), callback=lambda sys: sys.version_info)
119 In [3]: test_for('sys', (2,6), callback=lambda sys: sys.version_info)
120 Out[3]: True
120 Out[3]: True
121
121
122 """
122 """
123 try:
123 try:
124 check = import_item(item)
124 check = import_item(item)
125 except (ImportError, RuntimeError):
125 except (ImportError, RuntimeError):
126 # GTK reports Runtime error if it can't be initialized even if it's
126 # GTK reports Runtime error if it can't be initialized even if it's
127 # importable.
127 # importable.
128 return False
128 return False
129 else:
129 else:
130 if min_version:
130 if min_version:
131 if callback:
131 if callback:
132 # extra processing step to get version to compare
132 # extra processing step to get version to compare
133 check = callback(check)
133 check = callback(check)
134
134
135 return check >= min_version
135 return check >= min_version
136 else:
136 else:
137 return True
137 return True
138
138
139 # Global dict where we can store information on what we have and what we don't
139 # Global dict where we can store information on what we have and what we don't
140 # have available at test run time
140 # have available at test run time
141 have = {}
141 have = {}
142
142
143 have['curses'] = test_for('_curses')
143 have['curses'] = test_for('_curses')
144 have['matplotlib'] = test_for('matplotlib')
144 have['matplotlib'] = test_for('matplotlib')
145 have['pexpect'] = test_for('IPython.external.pexpect')
145 have['pexpect'] = test_for('IPython.external.pexpect')
146 have['pymongo'] = test_for('pymongo')
146 have['pymongo'] = test_for('pymongo')
147 have['wx'] = test_for('wx')
147 have['wx'] = test_for('wx')
148 have['wx.aui'] = test_for('wx.aui')
148 have['wx.aui'] = test_for('wx.aui')
149 have['qt'] = test_for('IPython.external.qt')
149 have['qt'] = test_for('IPython.external.qt')
150 have['sqlite3'] = test_for('sqlite3')
150 have['sqlite3'] = test_for('sqlite3')
151 have['cython'] = test_for('Cython')
151
152
152 have['tornado'] = test_for('tornado.version_info', (2,1,0), callback=None)
153 have['tornado'] = test_for('tornado.version_info', (2,1,0), callback=None)
153
154
154 if os.name == 'nt':
155 if os.name == 'nt':
155 min_zmq = (2,1,7)
156 min_zmq = (2,1,7)
156 else:
157 else:
157 min_zmq = (2,1,4)
158 min_zmq = (2,1,4)
158
159
159 def version_tuple(mod):
160 def version_tuple(mod):
160 "turn '2.1.9' into (2,1,9), and '2.1dev' into (2,1,999)"
161 "turn '2.1.9' into (2,1,9), and '2.1dev' into (2,1,999)"
161 # turn 'dev' into 999, because Python3 rejects str-int comparisons
162 # turn 'dev' into 999, because Python3 rejects str-int comparisons
162 vs = mod.__version__.replace('dev', '.999')
163 vs = mod.__version__.replace('dev', '.999')
163 tup = tuple([int(v) for v in vs.split('.') ])
164 tup = tuple([int(v) for v in vs.split('.') ])
164 return tup
165 return tup
165
166
166 have['zmq'] = test_for('zmq', min_zmq, version_tuple)
167 have['zmq'] = test_for('zmq', min_zmq, version_tuple)
167
168
168 #-----------------------------------------------------------------------------
169 #-----------------------------------------------------------------------------
169 # Functions and classes
170 # Functions and classes
170 #-----------------------------------------------------------------------------
171 #-----------------------------------------------------------------------------
171
172
172 def report():
173 def report():
173 """Return a string with a summary report of test-related variables."""
174 """Return a string with a summary report of test-related variables."""
174
175
175 out = [ sys_info(), '\n']
176 out = [ sys_info(), '\n']
176
177
177 avail = []
178 avail = []
178 not_avail = []
179 not_avail = []
179
180
180 for k, is_avail in have.items():
181 for k, is_avail in have.items():
181 if is_avail:
182 if is_avail:
182 avail.append(k)
183 avail.append(k)
183 else:
184 else:
184 not_avail.append(k)
185 not_avail.append(k)
185
186
186 if avail:
187 if avail:
187 out.append('\nTools and libraries available at test time:\n')
188 out.append('\nTools and libraries available at test time:\n')
188 avail.sort()
189 avail.sort()
189 out.append(' ' + ' '.join(avail)+'\n')
190 out.append(' ' + ' '.join(avail)+'\n')
190
191
191 if not_avail:
192 if not_avail:
192 out.append('\nTools and libraries NOT available at test time:\n')
193 out.append('\nTools and libraries NOT available at test time:\n')
193 not_avail.sort()
194 not_avail.sort()
194 out.append(' ' + ' '.join(not_avail)+'\n')
195 out.append(' ' + ' '.join(not_avail)+'\n')
195
196
196 return ''.join(out)
197 return ''.join(out)
197
198
198
199
199 def make_exclude():
200 def make_exclude():
200 """Make patterns of modules and packages to exclude from testing.
201 """Make patterns of modules and packages to exclude from testing.
201
202
202 For the IPythonDoctest plugin, we need to exclude certain patterns that
203 For the IPythonDoctest plugin, we need to exclude certain patterns that
203 cause testing problems. We should strive to minimize the number of
204 cause testing problems. We should strive to minimize the number of
204 skipped modules, since this means untested code.
205 skipped modules, since this means untested code.
205
206
206 These modules and packages will NOT get scanned by nose at all for tests.
207 These modules and packages will NOT get scanned by nose at all for tests.
207 """
208 """
208 # Simple utility to make IPython paths more readably, we need a lot of
209 # Simple utility to make IPython paths more readably, we need a lot of
209 # these below
210 # these below
210 ipjoin = lambda *paths: pjoin('IPython', *paths)
211 ipjoin = lambda *paths: pjoin('IPython', *paths)
211
212
212 exclusions = [ipjoin('external'),
213 exclusions = [ipjoin('external'),
213 pjoin('IPython_doctest_plugin'),
214 pjoin('IPython_doctest_plugin'),
214 ipjoin('quarantine'),
215 ipjoin('quarantine'),
215 ipjoin('deathrow'),
216 ipjoin('deathrow'),
216 ipjoin('testing', 'attic'),
217 ipjoin('testing', 'attic'),
217 # This guy is probably attic material
218 # This guy is probably attic material
218 ipjoin('testing', 'mkdoctests'),
219 ipjoin('testing', 'mkdoctests'),
219 # Testing inputhook will need a lot of thought, to figure out
220 # Testing inputhook will need a lot of thought, to figure out
220 # how to have tests that don't lock up with the gui event
221 # how to have tests that don't lock up with the gui event
221 # loops in the picture
222 # loops in the picture
222 ipjoin('lib', 'inputhook'),
223 ipjoin('lib', 'inputhook'),
223 # Config files aren't really importable stand-alone
224 # Config files aren't really importable stand-alone
224 ipjoin('config', 'default'),
225 ipjoin('config', 'default'),
225 ipjoin('config', 'profile'),
226 ipjoin('config', 'profile'),
226 ]
227 ]
227 if not have['sqlite3']:
228 if not have['sqlite3']:
228 exclusions.append(ipjoin('core', 'tests', 'test_history'))
229 exclusions.append(ipjoin('core', 'tests', 'test_history'))
229 exclusions.append(ipjoin('core', 'history'))
230 exclusions.append(ipjoin('core', 'history'))
230 if not have['wx']:
231 if not have['wx']:
231 exclusions.append(ipjoin('lib', 'inputhookwx'))
232 exclusions.append(ipjoin('lib', 'inputhookwx'))
232
233
233 # We do this unconditionally, so that the test suite doesn't import
234 # We do this unconditionally, so that the test suite doesn't import
234 # gtk, changing the default encoding and masking some unicode bugs.
235 # gtk, changing the default encoding and masking some unicode bugs.
235 exclusions.append(ipjoin('lib', 'inputhookgtk'))
236 exclusions.append(ipjoin('lib', 'inputhookgtk'))
236 exclusions.append(ipjoin('zmq', 'gui', 'gtkembed'))
237 exclusions.append(ipjoin('zmq', 'gui', 'gtkembed'))
237
238
238 # These have to be skipped on win32 because the use echo, rm, cd, etc.
239 # These have to be skipped on win32 because the use echo, rm, cd, etc.
239 # See ticket https://github.com/ipython/ipython/issues/87
240 # See ticket https://github.com/ipython/ipython/issues/87
240 if sys.platform == 'win32':
241 if sys.platform == 'win32':
241 exclusions.append(ipjoin('testing', 'plugin', 'test_exampleip'))
242 exclusions.append(ipjoin('testing', 'plugin', 'test_exampleip'))
242 exclusions.append(ipjoin('testing', 'plugin', 'dtexample'))
243 exclusions.append(ipjoin('testing', 'plugin', 'dtexample'))
243
244
244 if not have['pexpect']:
245 if not have['pexpect']:
245 exclusions.extend([ipjoin('scripts', 'irunner'),
246 exclusions.extend([ipjoin('scripts', 'irunner'),
246 ipjoin('lib', 'irunner'),
247 ipjoin('lib', 'irunner'),
247 ipjoin('lib', 'tests', 'test_irunner'),
248 ipjoin('lib', 'tests', 'test_irunner'),
248 ipjoin('frontend', 'terminal', 'console'),
249 ipjoin('frontend', 'terminal', 'console'),
249 ])
250 ])
250
251
251 if not have['zmq']:
252 if not have['zmq']:
252 exclusions.append(ipjoin('zmq'))
253 exclusions.append(ipjoin('zmq'))
253 exclusions.append(ipjoin('frontend', 'qt'))
254 exclusions.append(ipjoin('frontend', 'qt'))
254 exclusions.append(ipjoin('frontend', 'html'))
255 exclusions.append(ipjoin('frontend', 'html'))
255 exclusions.append(ipjoin('frontend', 'consoleapp.py'))
256 exclusions.append(ipjoin('frontend', 'consoleapp.py'))
256 exclusions.append(ipjoin('frontend', 'terminal', 'console'))
257 exclusions.append(ipjoin('frontend', 'terminal', 'console'))
257 exclusions.append(ipjoin('parallel'))
258 exclusions.append(ipjoin('parallel'))
258 elif not have['qt']:
259 elif not have['qt']:
259 exclusions.append(ipjoin('frontend', 'qt'))
260 exclusions.append(ipjoin('frontend', 'qt'))
260
261
261 if not have['pymongo']:
262 if not have['pymongo']:
262 exclusions.append(ipjoin('parallel', 'controller', 'mongodb'))
263 exclusions.append(ipjoin('parallel', 'controller', 'mongodb'))
263 exclusions.append(ipjoin('parallel', 'tests', 'test_mongodb'))
264 exclusions.append(ipjoin('parallel', 'tests', 'test_mongodb'))
264
265
265 if not have['matplotlib']:
266 if not have['matplotlib']:
266 exclusions.extend([ipjoin('core', 'pylabtools'),
267 exclusions.extend([ipjoin('core', 'pylabtools'),
267 ipjoin('core', 'tests', 'test_pylabtools'),
268 ipjoin('core', 'tests', 'test_pylabtools'),
268 ipjoin('zmq', 'pylab'),
269 ipjoin('zmq', 'pylab'),
269 ])
270 ])
270
271
272 if not have['cython']:
273 exclusions.extend([ipjoin('extensions', 'cythonmagic')])
274
271 if not have['tornado']:
275 if not have['tornado']:
272 exclusions.append(ipjoin('frontend', 'html'))
276 exclusions.append(ipjoin('frontend', 'html'))
273
277
274 # This is needed for the reg-exp to match on win32 in the ipdoctest plugin.
278 # This is needed for the reg-exp to match on win32 in the ipdoctest plugin.
275 if sys.platform == 'win32':
279 if sys.platform == 'win32':
276 exclusions = [s.replace('\\','\\\\') for s in exclusions]
280 exclusions = [s.replace('\\','\\\\') for s in exclusions]
277
281
278 return exclusions
282 return exclusions
279
283
280
284
281 class IPTester(object):
285 class IPTester(object):
282 """Call that calls iptest or trial in a subprocess.
286 """Call that calls iptest or trial in a subprocess.
283 """
287 """
284 #: string, name of test runner that will be called
288 #: string, name of test runner that will be called
285 runner = None
289 runner = None
286 #: list, parameters for test runner
290 #: list, parameters for test runner
287 params = None
291 params = None
288 #: list, arguments of system call to be made to call test runner
292 #: list, arguments of system call to be made to call test runner
289 call_args = None
293 call_args = None
290 #: list, process ids of subprocesses we start (for cleanup)
294 #: list, process ids of subprocesses we start (for cleanup)
291 pids = None
295 pids = None
292 #: str, coverage xml output file
296 #: str, coverage xml output file
293 coverage_xml = None
297 coverage_xml = None
294
298
295 def __init__(self, runner='iptest', params=None):
299 def __init__(self, runner='iptest', params=None):
296 """Create new test runner."""
300 """Create new test runner."""
297 p = os.path
301 p = os.path
298 if runner == 'iptest':
302 if runner == 'iptest':
299 iptest_app = get_ipython_module_path('IPython.testing.iptest')
303 iptest_app = get_ipython_module_path('IPython.testing.iptest')
300 self.runner = pycmd2argv(iptest_app) + sys.argv[1:]
304 self.runner = pycmd2argv(iptest_app) + sys.argv[1:]
301 else:
305 else:
302 raise Exception('Not a valid test runner: %s' % repr(runner))
306 raise Exception('Not a valid test runner: %s' % repr(runner))
303 if params is None:
307 if params is None:
304 params = []
308 params = []
305 if isinstance(params, str):
309 if isinstance(params, str):
306 params = [params]
310 params = [params]
307 self.params = params
311 self.params = params
308
312
309 # Assemble call
313 # Assemble call
310 self.call_args = self.runner+self.params
314 self.call_args = self.runner+self.params
311
315
312 # Find the section we're testing (IPython.foo)
316 # Find the section we're testing (IPython.foo)
313 for sect in self.params:
317 for sect in self.params:
314 if sect.startswith('IPython'): break
318 if sect.startswith('IPython'): break
315 else:
319 else:
316 raise ValueError("Section not found", self.params)
320 raise ValueError("Section not found", self.params)
317
321
318 if '--with-xunit' in self.call_args:
322 if '--with-xunit' in self.call_args:
319 self.call_args.append('--xunit-file=%s' % path.abspath(sect+'.xunit.xml'))
323 self.call_args.append('--xunit-file=%s' % path.abspath(sect+'.xunit.xml'))
320
324
321 if '--with-xml-coverage' in self.call_args:
325 if '--with-xml-coverage' in self.call_args:
322 self.coverage_xml = path.abspath(sect+".coverage.xml")
326 self.coverage_xml = path.abspath(sect+".coverage.xml")
323 self.call_args.remove('--with-xml-coverage')
327 self.call_args.remove('--with-xml-coverage')
324 self.call_args = ["coverage", "run", "--source="+sect] + self.call_args[1:]
328 self.call_args = ["coverage", "run", "--source="+sect] + self.call_args[1:]
325
329
326 # Store pids of anything we start to clean up on deletion, if possible
330 # Store pids of anything we start to clean up on deletion, if possible
327 # (on posix only, since win32 has no os.kill)
331 # (on posix only, since win32 has no os.kill)
328 self.pids = []
332 self.pids = []
329
333
330 if sys.platform == 'win32':
334 if sys.platform == 'win32':
331 def _run_cmd(self):
335 def _run_cmd(self):
332 # On Windows, use os.system instead of subprocess.call, because I
336 # On Windows, use os.system instead of subprocess.call, because I
333 # was having problems with subprocess and I just don't know enough
337 # was having problems with subprocess and I just don't know enough
334 # about win32 to debug this reliably. Os.system may be the 'old
338 # about win32 to debug this reliably. Os.system may be the 'old
335 # fashioned' way to do it, but it works just fine. If someone
339 # fashioned' way to do it, but it works just fine. If someone
336 # later can clean this up that's fine, as long as the tests run
340 # later can clean this up that's fine, as long as the tests run
337 # reliably in win32.
341 # reliably in win32.
338 # What types of problems are you having. They may be related to
342 # What types of problems are you having. They may be related to
339 # running Python in unboffered mode. BG.
343 # running Python in unboffered mode. BG.
340 return os.system(' '.join(self.call_args))
344 return os.system(' '.join(self.call_args))
341 else:
345 else:
342 def _run_cmd(self):
346 def _run_cmd(self):
343 # print >> sys.stderr, '*** CMD:', ' '.join(self.call_args) # dbg
347 # print >> sys.stderr, '*** CMD:', ' '.join(self.call_args) # dbg
344 subp = subprocess.Popen(self.call_args)
348 subp = subprocess.Popen(self.call_args)
345 self.pids.append(subp.pid)
349 self.pids.append(subp.pid)
346 # If this fails, the pid will be left in self.pids and cleaned up
350 # If this fails, the pid will be left in self.pids and cleaned up
347 # later, but if the wait call succeeds, then we can clear the
351 # later, but if the wait call succeeds, then we can clear the
348 # stored pid.
352 # stored pid.
349 retcode = subp.wait()
353 retcode = subp.wait()
350 self.pids.pop()
354 self.pids.pop()
351 return retcode
355 return retcode
352
356
353 def run(self):
357 def run(self):
354 """Run the stored commands"""
358 """Run the stored commands"""
355 try:
359 try:
356 retcode = self._run_cmd()
360 retcode = self._run_cmd()
357 except:
361 except:
358 import traceback
362 import traceback
359 traceback.print_exc()
363 traceback.print_exc()
360 return 1 # signal failure
364 return 1 # signal failure
361
365
362 if self.coverage_xml:
366 if self.coverage_xml:
363 subprocess.call(["coverage", "xml", "-o", self.coverage_xml])
367 subprocess.call(["coverage", "xml", "-o", self.coverage_xml])
364 return retcode
368 return retcode
365
369
366 def __del__(self):
370 def __del__(self):
367 """Cleanup on exit by killing any leftover processes."""
371 """Cleanup on exit by killing any leftover processes."""
368
372
369 if not hasattr(os, 'kill'):
373 if not hasattr(os, 'kill'):
370 return
374 return
371
375
372 for pid in self.pids:
376 for pid in self.pids:
373 try:
377 try:
374 print 'Cleaning stale PID:', pid
378 print 'Cleaning stale PID:', pid
375 os.kill(pid, signal.SIGKILL)
379 os.kill(pid, signal.SIGKILL)
376 except OSError:
380 except OSError:
377 # This is just a best effort, if we fail or the process was
381 # This is just a best effort, if we fail or the process was
378 # really gone, ignore it.
382 # really gone, ignore it.
379 pass
383 pass
380
384
381
385
382 def make_runners():
386 def make_runners():
383 """Define the top-level packages that need to be tested.
387 """Define the top-level packages that need to be tested.
384 """
388 """
385
389
386 # Packages to be tested via nose, that only depend on the stdlib
390 # Packages to be tested via nose, that only depend on the stdlib
387 nose_pkg_names = ['config', 'core', 'extensions', 'frontend', 'lib',
391 nose_pkg_names = ['config', 'core', 'extensions', 'frontend', 'lib',
388 'scripts', 'testing', 'utils', 'nbformat' ]
392 'scripts', 'testing', 'utils', 'nbformat' ]
389
393
390 if have['zmq']:
394 if have['zmq']:
391 nose_pkg_names.append('zmq')
395 nose_pkg_names.append('zmq')
392 nose_pkg_names.append('parallel')
396 nose_pkg_names.append('parallel')
393
397
394 # For debugging this code, only load quick stuff
398 # For debugging this code, only load quick stuff
395 #nose_pkg_names = ['core', 'extensions'] # dbg
399 #nose_pkg_names = ['core', 'extensions'] # dbg
396
400
397 # Make fully qualified package names prepending 'IPython.' to our name lists
401 # Make fully qualified package names prepending 'IPython.' to our name lists
398 nose_packages = ['IPython.%s' % m for m in nose_pkg_names ]
402 nose_packages = ['IPython.%s' % m for m in nose_pkg_names ]
399
403
400 # Make runners
404 # Make runners
401 runners = [ (v, IPTester('iptest', params=v)) for v in nose_packages ]
405 runners = [ (v, IPTester('iptest', params=v)) for v in nose_packages ]
402
406
403 return runners
407 return runners
404
408
405
409
406 def run_iptest():
410 def run_iptest():
407 """Run the IPython test suite using nose.
411 """Run the IPython test suite using nose.
408
412
409 This function is called when this script is **not** called with the form
413 This function is called when this script is **not** called with the form
410 `iptest all`. It simply calls nose with appropriate command line flags
414 `iptest all`. It simply calls nose with appropriate command line flags
411 and accepts all of the standard nose arguments.
415 and accepts all of the standard nose arguments.
412 """
416 """
413 # Apply our monkeypatch to Xunit
417 # Apply our monkeypatch to Xunit
414 if '--with-xunit' in sys.argv and not hasattr(Xunit, 'orig_addError'):
418 if '--with-xunit' in sys.argv and not hasattr(Xunit, 'orig_addError'):
415 monkeypatch_xunit()
419 monkeypatch_xunit()
416
420
417 warnings.filterwarnings('ignore',
421 warnings.filterwarnings('ignore',
418 'This will be removed soon. Use IPython.testing.util instead')
422 'This will be removed soon. Use IPython.testing.util instead')
419
423
420 argv = sys.argv + [ '--detailed-errors', # extra info in tracebacks
424 argv = sys.argv + [ '--detailed-errors', # extra info in tracebacks
421
425
422 '--with-ipdoctest',
426 '--with-ipdoctest',
423 '--ipdoctest-tests','--ipdoctest-extension=txt',
427 '--ipdoctest-tests','--ipdoctest-extension=txt',
424
428
425 # We add --exe because of setuptools' imbecility (it
429 # We add --exe because of setuptools' imbecility (it
426 # blindly does chmod +x on ALL files). Nose does the
430 # blindly does chmod +x on ALL files). Nose does the
427 # right thing and it tries to avoid executables,
431 # right thing and it tries to avoid executables,
428 # setuptools unfortunately forces our hand here. This
432 # setuptools unfortunately forces our hand here. This
429 # has been discussed on the distutils list and the
433 # has been discussed on the distutils list and the
430 # setuptools devs refuse to fix this problem!
434 # setuptools devs refuse to fix this problem!
431 '--exe',
435 '--exe',
432 ]
436 ]
433
437
434 if nose.__version__ >= '0.11':
438 if nose.__version__ >= '0.11':
435 # I don't fully understand why we need this one, but depending on what
439 # I don't fully understand why we need this one, but depending on what
436 # directory the test suite is run from, if we don't give it, 0 tests
440 # directory the test suite is run from, if we don't give it, 0 tests
437 # get run. Specifically, if the test suite is run from the source dir
441 # get run. Specifically, if the test suite is run from the source dir
438 # with an argument (like 'iptest.py IPython.core', 0 tests are run,
442 # with an argument (like 'iptest.py IPython.core', 0 tests are run,
439 # even if the same call done in this directory works fine). It appears
443 # even if the same call done in this directory works fine). It appears
440 # that if the requested package is in the current dir, nose bails early
444 # that if the requested package is in the current dir, nose bails early
441 # by default. Since it's otherwise harmless, leave it in by default
445 # by default. Since it's otherwise harmless, leave it in by default
442 # for nose >= 0.11, though unfortunately nose 0.10 doesn't support it.
446 # for nose >= 0.11, though unfortunately nose 0.10 doesn't support it.
443 argv.append('--traverse-namespace')
447 argv.append('--traverse-namespace')
444
448
445 # use our plugin for doctesting. It will remove the standard doctest plugin
449 # use our plugin for doctesting. It will remove the standard doctest plugin
446 # if it finds it enabled
450 # if it finds it enabled
447 plugins = [IPythonDoctest(make_exclude()), KnownFailure()]
451 plugins = [IPythonDoctest(make_exclude()), KnownFailure()]
448 # We need a global ipython running in this process
452 # We need a global ipython running in this process
449 globalipapp.start_ipython()
453 globalipapp.start_ipython()
450 # Now nose can run
454 # Now nose can run
451 TestProgram(argv=argv, addplugins=plugins)
455 TestProgram(argv=argv, addplugins=plugins)
452
456
453
457
454 def run_iptestall():
458 def run_iptestall():
455 """Run the entire IPython test suite by calling nose and trial.
459 """Run the entire IPython test suite by calling nose and trial.
456
460
457 This function constructs :class:`IPTester` instances for all IPython
461 This function constructs :class:`IPTester` instances for all IPython
458 modules and package and then runs each of them. This causes the modules
462 modules and package and then runs each of them. This causes the modules
459 and packages of IPython to be tested each in their own subprocess using
463 and packages of IPython to be tested each in their own subprocess using
460 nose.
464 nose.
461 """
465 """
462
466
463 runners = make_runners()
467 runners = make_runners()
464
468
465 # Run the test runners in a temporary dir so we can nuke it when finished
469 # Run the test runners in a temporary dir so we can nuke it when finished
466 # to clean up any junk files left over by accident. This also makes it
470 # to clean up any junk files left over by accident. This also makes it
467 # robust against being run in non-writeable directories by mistake, as the
471 # robust against being run in non-writeable directories by mistake, as the
468 # temp dir will always be user-writeable.
472 # temp dir will always be user-writeable.
469 curdir = os.getcwdu()
473 curdir = os.getcwdu()
470 testdir = tempfile.gettempdir()
474 testdir = tempfile.gettempdir()
471 os.chdir(testdir)
475 os.chdir(testdir)
472
476
473 # Run all test runners, tracking execution time
477 # Run all test runners, tracking execution time
474 failed = []
478 failed = []
475 t_start = time.time()
479 t_start = time.time()
476 try:
480 try:
477 for (name, runner) in runners:
481 for (name, runner) in runners:
478 print '*'*70
482 print '*'*70
479 print 'IPython test group:',name
483 print 'IPython test group:',name
480 res = runner.run()
484 res = runner.run()
481 if res:
485 if res:
482 failed.append( (name, runner) )
486 failed.append( (name, runner) )
483 finally:
487 finally:
484 os.chdir(curdir)
488 os.chdir(curdir)
485 t_end = time.time()
489 t_end = time.time()
486 t_tests = t_end - t_start
490 t_tests = t_end - t_start
487 nrunners = len(runners)
491 nrunners = len(runners)
488 nfail = len(failed)
492 nfail = len(failed)
489 # summarize results
493 # summarize results
490 print
494 print
491 print '*'*70
495 print '*'*70
492 print 'Test suite completed for system with the following information:'
496 print 'Test suite completed for system with the following information:'
493 print report()
497 print report()
494 print 'Ran %s test groups in %.3fs' % (nrunners, t_tests)
498 print 'Ran %s test groups in %.3fs' % (nrunners, t_tests)
495 print
499 print
496 print 'Status:'
500 print 'Status:'
497 if not failed:
501 if not failed:
498 print 'OK'
502 print 'OK'
499 else:
503 else:
500 # If anything went wrong, point out what command to rerun manually to
504 # If anything went wrong, point out what command to rerun manually to
501 # see the actual errors and individual summary
505 # see the actual errors and individual summary
502 print 'ERROR - %s out of %s test groups failed.' % (nfail, nrunners)
506 print 'ERROR - %s out of %s test groups failed.' % (nfail, nrunners)
503 for name, failed_runner in failed:
507 for name, failed_runner in failed:
504 print '-'*40
508 print '-'*40
505 print 'Runner failed:',name
509 print 'Runner failed:',name
506 print 'You may wish to rerun this one individually, with:'
510 print 'You may wish to rerun this one individually, with:'
507 print ' '.join(failed_runner.call_args)
511 print ' '.join(failed_runner.call_args)
508 print
512 print
509 # Ensure that our exit code indicates failure
513 # Ensure that our exit code indicates failure
510 sys.exit(1)
514 sys.exit(1)
511
515
512
516
513 def main():
517 def main():
514 for arg in sys.argv[1:]:
518 for arg in sys.argv[1:]:
515 if arg.startswith('IPython'):
519 if arg.startswith('IPython'):
516 # This is in-process
520 # This is in-process
517 run_iptest()
521 run_iptest()
518 else:
522 else:
519 # This starts subprocesses
523 # This starts subprocesses
520 run_iptestall()
524 run_iptestall()
521
525
522
526
523 if __name__ == '__main__':
527 if __name__ == '__main__':
524 main()
528 main()
@@ -1,76 +1,77
1 .. _extensions_overview:
1 .. _extensions_overview:
2
2
3 ==================
3 ==================
4 IPython extensions
4 IPython extensions
5 ==================
5 ==================
6
6
7 Configuration files are just the first level of customization that IPython
7 Configuration files are just the first level of customization that IPython
8 supports. The next level is that of extensions. An IPython extension is an
8 supports. The next level is that of extensions. An IPython extension is an
9 importable Python module that has a a few special function. By defining these
9 importable Python module that has a a few special function. By defining these
10 functions, users can customize IPython by accessing the actual runtime objects
10 functions, users can customize IPython by accessing the actual runtime objects
11 of IPython. Here is a sample extension::
11 of IPython. Here is a sample extension::
12
12
13 # myextension.py
13 # myextension.py
14
14
15 def load_ipython_extension(ipython):
15 def load_ipython_extension(ipython):
16 # The ``ipython`` argument is the currently active
16 # The ``ipython`` argument is the currently active
17 # :class:`InteractiveShell` instance that can be used in any way.
17 # :class:`InteractiveShell` instance that can be used in any way.
18 # This allows you do to things like register new magics, plugins or
18 # This allows you do to things like register new magics, plugins or
19 # aliases.
19 # aliases.
20
20
21 def unload_ipython_extension(ipython):
21 def unload_ipython_extension(ipython):
22 # If you want your extension to be unloadable, put that logic here.
22 # If you want your extension to be unloadable, put that logic here.
23
23
24 This :func:`load_ipython_extension` function is called after your extension is
24 This :func:`load_ipython_extension` function is called after your extension is
25 imported and the currently active :class:`InteractiveShell` instance is passed
25 imported and the currently active :class:`InteractiveShell` instance is passed
26 as the only argument. You can do anything you want with IPython at that point.
26 as the only argument. You can do anything you want with IPython at that point.
27
27
28 The :func:`load_ipython_extension` will be called again is you load or reload
28 The :func:`load_ipython_extension` will be called again is you load or reload
29 the extension again. It is up to the extension author to add code to manage
29 the extension again. It is up to the extension author to add code to manage
30 that.
30 that.
31
31
32 Useful :class:`InteractiveShell` methods include :meth:`~IPython.core.interactiveshell.InteractiveShell.define_magic`,
32 Useful :class:`InteractiveShell` methods include :meth:`~IPython.core.interactiveshell.InteractiveShell.define_magic`,
33 :meth:`~IPython.core.interactiveshell.InteractiveShell.push` (to add variables to the user namespace) and
33 :meth:`~IPython.core.interactiveshell.InteractiveShell.push` (to add variables to the user namespace) and
34 :meth:`~IPython.core.interactiveshell.InteractiveShell.drop_by_id` (to remove variables on unloading).
34 :meth:`~IPython.core.interactiveshell.InteractiveShell.drop_by_id` (to remove variables on unloading).
35
35
36 You can put your extension modules anywhere you want, as long as they can be
36 You can put your extension modules anywhere you want, as long as they can be
37 imported by Python's standard import mechanism. However, to make it easy to
37 imported by Python's standard import mechanism. However, to make it easy to
38 write extensions, you can also put your extensions in
38 write extensions, you can also put your extensions in
39 ``os.path.join(self.ipython_dir, 'extensions')``. This directory is added to
39 ``os.path.join(self.ipython_dir, 'extensions')``. This directory is added to
40 ``sys.path`` automatically.
40 ``sys.path`` automatically.
41
41
42 Using extensions
42 Using extensions
43 ================
43 ================
44
44
45 There are two ways you can tell IPython to use your extension:
45 There are two ways you can tell IPython to use your extension:
46
46
47 1. Listing it in a configuration file.
47 1. Listing it in a configuration file.
48 2. Using the ``%load_ext`` magic function.
48 2. Using the ``%load_ext`` magic function.
49
49
50 To load an extension called :file:`myextension.py` add the following logic
50 To load an extension called :file:`myextension.py` add the following logic
51 to your configuration file::
51 to your configuration file::
52
52
53 c.InteractiveShellApp.extensions = [
53 c.InteractiveShellApp.extensions = [
54 'myextension'
54 'myextension'
55 ]
55 ]
56
56
57 To load that same extension at runtime, use the ``%load_ext`` magic:
57 To load that same extension at runtime, use the ``%load_ext`` magic:
58
58
59 .. sourcecode:: ipython
59 .. sourcecode:: ipython
60
60
61 In [1]: %load_ext myextension
61 In [1]: %load_ext myextension
62
62
63 To summarize, in conjunction with configuration files and profiles, IPython
63 To summarize, in conjunction with configuration files and profiles, IPython
64 extensions give you complete and flexible control over your IPython
64 extensions give you complete and flexible control over your IPython
65 setup.
65 setup.
66
66
67 Extensions bundled with IPython
67 Extensions bundled with IPython
68 ===============================
68 ===============================
69
69
70 .. toctree::
70 .. toctree::
71 :maxdepth: 1
71 :maxdepth: 1
72
72
73 autoreload
73 autoreload
74 cythonmagic
74 parallelmagic
75 parallelmagic
75 storemagic
76 storemagic
76 sympyprinting
77 sympyprinting
General Comments 0
You need to be logged in to leave comments. Login now