##// END OF EJS Templates
Merge pull request #13348 from minrk/run-async-tests...
Matthias Bussonnier -
r27389:f162b26d merge
parent child Browse files
Show More
@@ -1,72 +1,87 b''
1 import types
2 import sys
3 import builtins
1 import builtins
2 import inspect
4 import os
3 import os
5 import pytest
6 import pathlib
4 import pathlib
7 import shutil
5 import shutil
6 import sys
7 import types
8
9 import pytest
8
10
9 # Must register before it gets imported
11 # Must register before it gets imported
10 pytest.register_assert_rewrite("IPython.testing.tools")
12 pytest.register_assert_rewrite("IPython.testing.tools")
11
13
12 from .testing import tools
14 from .testing import tools
13
15
14
16
17 def pytest_collection_modifyitems(items):
18 """This function is automatically run by pytest passing all collected test
19 functions.
20
21 We use it to add asyncio marker to all async tests and assert we don't use
22 test functions that are async generators which wouldn't make sense.
23 """
24 for item in items:
25 if inspect.iscoroutinefunction(item.obj):
26 item.add_marker("asyncio")
27 assert not inspect.isasyncgenfunction(item.obj)
28
29
15 def get_ipython():
30 def get_ipython():
16 from .terminal.interactiveshell import TerminalInteractiveShell
31 from .terminal.interactiveshell import TerminalInteractiveShell
17 if TerminalInteractiveShell._instance:
32 if TerminalInteractiveShell._instance:
18 return TerminalInteractiveShell.instance()
33 return TerminalInteractiveShell.instance()
19
34
20 config = tools.default_config()
35 config = tools.default_config()
21 config.TerminalInteractiveShell.simple_prompt = True
36 config.TerminalInteractiveShell.simple_prompt = True
22
37
23 # Create and initialize our test-friendly IPython instance.
38 # Create and initialize our test-friendly IPython instance.
24 shell = TerminalInteractiveShell.instance(config=config)
39 shell = TerminalInteractiveShell.instance(config=config)
25 return shell
40 return shell
26
41
27
42
28 @pytest.fixture(scope='session', autouse=True)
43 @pytest.fixture(scope='session', autouse=True)
29 def work_path():
44 def work_path():
30 path = pathlib.Path("./tmp-ipython-pytest-profiledir")
45 path = pathlib.Path("./tmp-ipython-pytest-profiledir")
31 os.environ["IPYTHONDIR"] = str(path.absolute())
46 os.environ["IPYTHONDIR"] = str(path.absolute())
32 if path.exists():
47 if path.exists():
33 raise ValueError('IPython dir temporary path already exists ! Did previous test run exit successfully ?')
48 raise ValueError('IPython dir temporary path already exists ! Did previous test run exit successfully ?')
34 path.mkdir()
49 path.mkdir()
35 yield
50 yield
36 shutil.rmtree(str(path.resolve()))
51 shutil.rmtree(str(path.resolve()))
37
52
38
53
39 def nopage(strng, start=0, screen_lines=0, pager_cmd=None):
54 def nopage(strng, start=0, screen_lines=0, pager_cmd=None):
40 if isinstance(strng, dict):
55 if isinstance(strng, dict):
41 strng = strng.get("text/plain", "")
56 strng = strng.get("text/plain", "")
42 print(strng)
57 print(strng)
43
58
44
59
45 def xsys(self, cmd):
60 def xsys(self, cmd):
46 """Replace the default system call with a capturing one for doctest.
61 """Replace the default system call with a capturing one for doctest.
47 """
62 """
48 # We use getoutput, but we need to strip it because pexpect captures
63 # We use getoutput, but we need to strip it because pexpect captures
49 # the trailing newline differently from commands.getoutput
64 # the trailing newline differently from commands.getoutput
50 print(self.getoutput(cmd, split=False, depth=1).rstrip(), end="", file=sys.stdout)
65 print(self.getoutput(cmd, split=False, depth=1).rstrip(), end="", file=sys.stdout)
51 sys.stdout.flush()
66 sys.stdout.flush()
52
67
53
68
54 # for things to work correctly we would need this as a session fixture;
69 # for things to work correctly we would need this as a session fixture;
55 # unfortunately this will fail on some test that get executed as _collection_
70 # unfortunately this will fail on some test that get executed as _collection_
56 # time (before the fixture run), in particular parametrized test that contain
71 # time (before the fixture run), in particular parametrized test that contain
57 # yields. so for now execute at import time.
72 # yields. so for now execute at import time.
58 #@pytest.fixture(autouse=True, scope='session')
73 #@pytest.fixture(autouse=True, scope='session')
59 def inject():
74 def inject():
60
75
61 builtins.get_ipython = get_ipython
76 builtins.get_ipython = get_ipython
62 builtins._ip = get_ipython()
77 builtins._ip = get_ipython()
63 builtins.ip = get_ipython()
78 builtins.ip = get_ipython()
64 builtins.ip.system = types.MethodType(xsys, ip)
79 builtins.ip.system = types.MethodType(xsys, ip)
65 builtins.ip.builtin_trap.activate()
80 builtins.ip.builtin_trap.activate()
66 from .core import page
81 from .core import page
67
82
68 page.pager_page = nopage
83 page.pager_page = nopage
69 # yield
84 # yield
70
85
71
86
72 inject()
87 inject()
@@ -1,103 +1,156 b''
1 """
1 """
2 Async helper function that are invalid syntax on Python 3.5 and below.
2 Async helper function that are invalid syntax on Python 3.5 and below.
3
3
4 This code is best effort, and may have edge cases not behaving as expected. In
4 This code is best effort, and may have edge cases not behaving as expected. In
5 particular it contain a number of heuristics to detect whether code is
5 particular it contain a number of heuristics to detect whether code is
6 effectively async and need to run in an event loop or not.
6 effectively async and need to run in an event loop or not.
7
7
8 Some constructs (like top-level `return`, or `yield`) are taken care of
8 Some constructs (like top-level `return`, or `yield`) are taken care of
9 explicitly to actually raise a SyntaxError and stay as close as possible to
9 explicitly to actually raise a SyntaxError and stay as close as possible to
10 Python semantics.
10 Python semantics.
11 """
11 """
12
12
13
13
14 import ast
14 import ast
15 import asyncio
15 import asyncio
16 import inspect
16 import inspect
17 from functools import wraps
17
18
19 _asyncio_event_loop = None
18
20
19 class _AsyncIORunner:
20 def __init__(self):
21 self._loop = None
22
23 @property
24 def loop(self):
25 """Always returns a non-closed event loop"""
26 if self._loop is None or self._loop.is_closed():
27 policy = asyncio.get_event_loop_policy()
28 self._loop = policy.new_event_loop()
29 policy.set_event_loop(self._loop)
30 return self._loop
31
21
22 def get_asyncio_loop():
23 """asyncio has deprecated get_event_loop
24
25 Replicate it here, with our desired semantics:
26
27 - always returns a valid, not-closed loop
28 - not thread-local like asyncio's,
29 because we only want one loop for IPython
30 - if called from inside a coroutine (e.g. in ipykernel),
31 return the running loop
32
33 .. versionadded:: 8.0
34 """
35 try:
36 return asyncio.get_running_loop()
37 except RuntimeError:
38 # not inside a coroutine,
39 # track our own global
40 pass
41
42 # not thread-local like asyncio's,
43 # because we only track one event loop to run for IPython itself,
44 # always in the main thread.
45 global _asyncio_event_loop
46 if _asyncio_event_loop is None or _asyncio_event_loop.is_closed():
47 _asyncio_event_loop = asyncio.new_event_loop()
48 return _asyncio_event_loop
49
50
51 class _AsyncIORunner:
32 def __call__(self, coro):
52 def __call__(self, coro):
33 """
53 """
34 Handler for asyncio autoawait
54 Handler for asyncio autoawait
35 """
55 """
36 return self.loop.run_until_complete(coro)
56 return get_asyncio_loop().run_until_complete(coro)
37
57
38 def __str__(self):
58 def __str__(self):
39 return "asyncio"
59 return "asyncio"
40
60
41
61
42 _asyncio_runner = _AsyncIORunner()
62 _asyncio_runner = _AsyncIORunner()
43
63
44
64
65 class _AsyncIOProxy:
66 """Proxy-object for an asyncio
67
68 Any coroutine methods will be wrapped in event_loop.run_
69 """
70
71 def __init__(self, obj, event_loop):
72 self._obj = obj
73 self._event_loop = event_loop
74
75 def __repr__(self):
76 return f"<_AsyncIOProxy({self._obj!r})>"
77
78 def __getattr__(self, key):
79 attr = getattr(self._obj, key)
80 if inspect.iscoroutinefunction(attr):
81 # if it's a coroutine method,
82 # return a threadsafe wrapper onto the _current_ asyncio loop
83 @wraps(attr)
84 def _wrapped(*args, **kwargs):
85 concurrent_future = asyncio.run_coroutine_threadsafe(
86 attr(*args, **kwargs), self._event_loop
87 )
88 return asyncio.wrap_future(concurrent_future)
89
90 return _wrapped
91 else:
92 return attr
93
94 def __dir__(self):
95 return dir(self._obj)
96
97
45 def _curio_runner(coroutine):
98 def _curio_runner(coroutine):
46 """
99 """
47 handler for curio autoawait
100 handler for curio autoawait
48 """
101 """
49 import curio
102 import curio
50
103
51 return curio.run(coroutine)
104 return curio.run(coroutine)
52
105
53
106
54 def _trio_runner(async_fn):
107 def _trio_runner(async_fn):
55 import trio
108 import trio
56
109
57 async def loc(coro):
110 async def loc(coro):
58 """
111 """
59 We need the dummy no-op async def to protect from
112 We need the dummy no-op async def to protect from
60 trio's internal. See https://github.com/python-trio/trio/issues/89
113 trio's internal. See https://github.com/python-trio/trio/issues/89
61 """
114 """
62 return await coro
115 return await coro
63
116
64 return trio.run(loc, async_fn)
117 return trio.run(loc, async_fn)
65
118
66
119
67 def _pseudo_sync_runner(coro):
120 def _pseudo_sync_runner(coro):
68 """
121 """
69 A runner that does not really allow async execution, and just advance the coroutine.
122 A runner that does not really allow async execution, and just advance the coroutine.
70
123
71 See discussion in https://github.com/python-trio/trio/issues/608,
124 See discussion in https://github.com/python-trio/trio/issues/608,
72
125
73 Credit to Nathaniel Smith
126 Credit to Nathaniel Smith
74 """
127 """
75 try:
128 try:
76 coro.send(None)
129 coro.send(None)
77 except StopIteration as exc:
130 except StopIteration as exc:
78 return exc.value
131 return exc.value
79 else:
132 else:
80 # TODO: do not raise but return an execution result with the right info.
133 # TODO: do not raise but return an execution result with the right info.
81 raise RuntimeError(
134 raise RuntimeError(
82 "{coro_name!r} needs a real async loop".format(coro_name=coro.__name__)
135 "{coro_name!r} needs a real async loop".format(coro_name=coro.__name__)
83 )
136 )
84
137
85
138
86 def _should_be_async(cell: str) -> bool:
139 def _should_be_async(cell: str) -> bool:
87 """Detect if a block of code need to be wrapped in an `async def`
140 """Detect if a block of code need to be wrapped in an `async def`
88
141
89 Attempt to parse the block of code, it it compile we're fine.
142 Attempt to parse the block of code, it it compile we're fine.
90 Otherwise we wrap if and try to compile.
143 Otherwise we wrap if and try to compile.
91
144
92 If it works, assume it should be async. Otherwise Return False.
145 If it works, assume it should be async. Otherwise Return False.
93
146
94 Not handled yet: If the block of code has a return statement as the top
147 Not handled yet: If the block of code has a return statement as the top
95 level, it will be seen as async. This is a know limitation.
148 level, it will be seen as async. This is a know limitation.
96 """
149 """
97 try:
150 try:
98 code = compile(
151 code = compile(
99 cell, "<>", "exec", flags=getattr(ast, "PyCF_ALLOW_TOP_LEVEL_AWAIT", 0x0)
152 cell, "<>", "exec", flags=getattr(ast, "PyCF_ALLOW_TOP_LEVEL_AWAIT", 0x0)
100 )
153 )
101 return inspect.CO_COROUTINE & code.co_flags == inspect.CO_COROUTINE
154 return inspect.CO_COROUTINE & code.co_flags == inspect.CO_COROUTINE
102 except (SyntaxError, MemoryError):
155 except (SyntaxError, MemoryError):
103 return False
156 return False
@@ -1,375 +1,362 b''
1 """Magic functions for running cells in various scripts."""
1 """Magic functions for running cells in various scripts."""
2
2
3 # Copyright (c) IPython Development Team.
3 # Copyright (c) IPython Development Team.
4 # Distributed under the terms of the Modified BSD License.
4 # Distributed under the terms of the Modified BSD License.
5
5
6 import asyncio
6 import asyncio
7 import atexit
7 import atexit
8 import errno
8 import errno
9 import functools
10 import os
9 import os
11 import signal
10 import signal
12 import sys
11 import sys
13 import time
12 import time
14 from contextlib import contextmanager
15 from subprocess import CalledProcessError
13 from subprocess import CalledProcessError
14 from threading import Thread
16
15
17 from traitlets import Dict, List, default
16 from traitlets import Any, Dict, List, default
18
17
19 from IPython.core import magic_arguments
18 from IPython.core import magic_arguments
19 from IPython.core.async_helpers import _AsyncIOProxy
20 from IPython.core.magic import Magics, cell_magic, line_magic, magics_class
20 from IPython.core.magic import Magics, cell_magic, line_magic, magics_class
21 from IPython.lib.backgroundjobs import BackgroundJobManager
22 from IPython.utils.process import arg_split
21 from IPython.utils.process import arg_split
23
22
24 #-----------------------------------------------------------------------------
23 #-----------------------------------------------------------------------------
25 # Magic implementation classes
24 # Magic implementation classes
26 #-----------------------------------------------------------------------------
25 #-----------------------------------------------------------------------------
27
26
28 def script_args(f):
27 def script_args(f):
29 """single decorator for adding script args"""
28 """single decorator for adding script args"""
30 args = [
29 args = [
31 magic_arguments.argument(
30 magic_arguments.argument(
32 '--out', type=str,
31 '--out', type=str,
33 help="""The variable in which to store stdout from the script.
32 help="""The variable in which to store stdout from the script.
34 If the script is backgrounded, this will be the stdout *pipe*,
33 If the script is backgrounded, this will be the stdout *pipe*,
35 instead of the stderr text itself and will not be auto closed.
34 instead of the stderr text itself and will not be auto closed.
36 """
35 """
37 ),
36 ),
38 magic_arguments.argument(
37 magic_arguments.argument(
39 '--err', type=str,
38 '--err', type=str,
40 help="""The variable in which to store stderr from the script.
39 help="""The variable in which to store stderr from the script.
41 If the script is backgrounded, this will be the stderr *pipe*,
40 If the script is backgrounded, this will be the stderr *pipe*,
42 instead of the stderr text itself and will not be autoclosed.
41 instead of the stderr text itself and will not be autoclosed.
43 """
42 """
44 ),
43 ),
45 magic_arguments.argument(
44 magic_arguments.argument(
46 '--bg', action="store_true",
45 '--bg', action="store_true",
47 help="""Whether to run the script in the background.
46 help="""Whether to run the script in the background.
48 If given, the only way to see the output of the command is
47 If given, the only way to see the output of the command is
49 with --out/err.
48 with --out/err.
50 """
49 """
51 ),
50 ),
52 magic_arguments.argument(
51 magic_arguments.argument(
53 '--proc', type=str,
52 '--proc', type=str,
54 help="""The variable in which to store Popen instance.
53 help="""The variable in which to store Popen instance.
55 This is used only when --bg option is given.
54 This is used only when --bg option is given.
56 """
55 """
57 ),
56 ),
58 magic_arguments.argument(
57 magic_arguments.argument(
59 '--no-raise-error', action="store_false", dest='raise_error',
58 '--no-raise-error', action="store_false", dest='raise_error',
60 help="""Whether you should raise an error message in addition to
59 help="""Whether you should raise an error message in addition to
61 a stream on stderr if you get a nonzero exit code.
60 a stream on stderr if you get a nonzero exit code.
62 """
61 """
63 )
62 )
64 ]
63 ]
65 for arg in args:
64 for arg in args:
66 f = arg(f)
65 f = arg(f)
67 return f
66 return f
68
67
69
68
70 @contextmanager
71 def safe_watcher():
72 if sys.platform == "win32":
73 yield
74 return
75
76 from asyncio import SafeChildWatcher
77
78 policy = asyncio.get_event_loop_policy()
79 old_watcher = policy.get_child_watcher()
80 if isinstance(old_watcher, SafeChildWatcher):
81 yield
82 return
83
84 try:
85 loop = policy.get_event_loop()
86 if loop.is_closed():
87 raise RuntimeError("open a new one")
88 except RuntimeError:
89 # closed loop, make a new one
90 loop = policy.new_event_loop()
91 policy.set_event_loop(loop)
92
93 try:
94 watcher = asyncio.SafeChildWatcher()
95 watcher.attach_loop(loop)
96 policy.set_child_watcher(watcher)
97 yield
98 finally:
99 watcher.close()
100 policy.set_child_watcher(old_watcher)
101
102
103 def dec_safe_watcher(fun):
104 @functools.wraps(fun)
105 def _inner(*args, **kwargs):
106 with safe_watcher():
107 return fun(*args, **kwargs)
108
109 return _inner
110
111
112 @magics_class
69 @magics_class
113 class ScriptMagics(Magics):
70 class ScriptMagics(Magics):
114 """Magics for talking to scripts
71 """Magics for talking to scripts
115
72
116 This defines a base `%%script` cell magic for running a cell
73 This defines a base `%%script` cell magic for running a cell
117 with a program in a subprocess, and registers a few top-level
74 with a program in a subprocess, and registers a few top-level
118 magics that call %%script with common interpreters.
75 magics that call %%script with common interpreters.
119 """
76 """
77
78 event_loop = Any(
79 help="""
80 The event loop on which to run subprocesses
81
82 Not the main event loop,
83 because we want to be able to make blocking calls
84 and have certain requirements we don't want to impose on the main loop.
85 """
86 )
87
120 script_magics = List(
88 script_magics = List(
121 help="""Extra script cell magics to define
89 help="""Extra script cell magics to define
122
90
123 This generates simple wrappers of `%%script foo` as `%%foo`.
91 This generates simple wrappers of `%%script foo` as `%%foo`.
124
92
125 If you want to add script magics that aren't on your path,
93 If you want to add script magics that aren't on your path,
126 specify them in script_paths
94 specify them in script_paths
127 """,
95 """,
128 ).tag(config=True)
96 ).tag(config=True)
129 @default('script_magics')
97 @default('script_magics')
130 def _script_magics_default(self):
98 def _script_magics_default(self):
131 """default to a common list of programs"""
99 """default to a common list of programs"""
132
100
133 defaults = [
101 defaults = [
134 'sh',
102 'sh',
135 'bash',
103 'bash',
136 'perl',
104 'perl',
137 'ruby',
105 'ruby',
138 'python',
106 'python',
139 'python2',
107 'python2',
140 'python3',
108 'python3',
141 'pypy',
109 'pypy',
142 ]
110 ]
143 if os.name == 'nt':
111 if os.name == 'nt':
144 defaults.extend([
112 defaults.extend([
145 'cmd',
113 'cmd',
146 ])
114 ])
147
115
148 return defaults
116 return defaults
149
117
150 script_paths = Dict(
118 script_paths = Dict(
151 help="""Dict mapping short 'ruby' names to full paths, such as '/opt/secret/bin/ruby'
119 help="""Dict mapping short 'ruby' names to full paths, such as '/opt/secret/bin/ruby'
152
120
153 Only necessary for items in script_magics where the default path will not
121 Only necessary for items in script_magics where the default path will not
154 find the right interpreter.
122 find the right interpreter.
155 """
123 """
156 ).tag(config=True)
124 ).tag(config=True)
157
125
158 def __init__(self, shell=None):
126 def __init__(self, shell=None):
159 super(ScriptMagics, self).__init__(shell=shell)
127 super(ScriptMagics, self).__init__(shell=shell)
160 self._generate_script_magics()
128 self._generate_script_magics()
161 self.job_manager = BackgroundJobManager()
162 self.bg_processes = []
129 self.bg_processes = []
163 atexit.register(self.kill_bg_processes)
130 atexit.register(self.kill_bg_processes)
164
131
165 def __del__(self):
132 def __del__(self):
166 self.kill_bg_processes()
133 self.kill_bg_processes()
167
134
168 def _generate_script_magics(self):
135 def _generate_script_magics(self):
169 cell_magics = self.magics['cell']
136 cell_magics = self.magics['cell']
170 for name in self.script_magics:
137 for name in self.script_magics:
171 cell_magics[name] = self._make_script_magic(name)
138 cell_magics[name] = self._make_script_magic(name)
172
139
173 def _make_script_magic(self, name):
140 def _make_script_magic(self, name):
174 """make a named magic, that calls %%script with a particular program"""
141 """make a named magic, that calls %%script with a particular program"""
175 # expand to explicit path if necessary:
142 # expand to explicit path if necessary:
176 script = self.script_paths.get(name, name)
143 script = self.script_paths.get(name, name)
177
144
178 @magic_arguments.magic_arguments()
145 @magic_arguments.magic_arguments()
179 @script_args
146 @script_args
180 def named_script_magic(line, cell):
147 def named_script_magic(line, cell):
181 # if line, add it as cl-flags
148 # if line, add it as cl-flags
182 if line:
149 if line:
183 line = "%s %s" % (script, line)
150 line = "%s %s" % (script, line)
184 else:
151 else:
185 line = script
152 line = script
186 return self.shebang(line, cell)
153 return self.shebang(line, cell)
187
154
188 # write a basic docstring:
155 # write a basic docstring:
189 named_script_magic.__doc__ = \
156 named_script_magic.__doc__ = \
190 """%%{name} script magic
157 """%%{name} script magic
191
158
192 Run cells with {script} in a subprocess.
159 Run cells with {script} in a subprocess.
193
160
194 This is a shortcut for `%%script {script}`
161 This is a shortcut for `%%script {script}`
195 """.format(**locals())
162 """.format(**locals())
196
163
197 return named_script_magic
164 return named_script_magic
198
165
199 @magic_arguments.magic_arguments()
166 @magic_arguments.magic_arguments()
200 @script_args
167 @script_args
201 @cell_magic("script")
168 @cell_magic("script")
202 @dec_safe_watcher
203 def shebang(self, line, cell):
169 def shebang(self, line, cell):
204 """Run a cell via a shell command
170 """Run a cell via a shell command
205
171
206 The `%%script` line is like the #! line of script,
172 The `%%script` line is like the #! line of script,
207 specifying a program (bash, perl, ruby, etc.) with which to run.
173 specifying a program (bash, perl, ruby, etc.) with which to run.
208
174
209 The rest of the cell is run by that program.
175 The rest of the cell is run by that program.
210
176
211 Examples
177 Examples
212 --------
178 --------
213 ::
179 ::
214
180
215 In [1]: %%script bash
181 In [1]: %%script bash
216 ...: for i in 1 2 3; do
182 ...: for i in 1 2 3; do
217 ...: echo $i
183 ...: echo $i
218 ...: done
184 ...: done
219 1
185 1
220 2
186 2
221 3
187 3
222 """
188 """
223
189
190 # Create the event loop in which to run script magics
191 # this operates on a background thread
192 if self.event_loop is None:
193 if sys.platform == "win32":
194 # don't override the current policy,
195 # just create an event loop
196 event_loop = asyncio.WindowsProactorEventLoopPolicy().new_event_loop()
197 else:
198 event_loop = asyncio.new_event_loop()
199 self.event_loop = event_loop
200
201 # start the loop in a background thread
202 asyncio_thread = Thread(target=event_loop.run_forever, daemon=True)
203 asyncio_thread.start()
204 else:
205 event_loop = self.event_loop
206
207 def in_thread(coro):
208 """Call a coroutine on the asyncio thread"""
209 return asyncio.run_coroutine_threadsafe(coro, event_loop).result()
210
224 async def _handle_stream(stream, stream_arg, file_object):
211 async def _handle_stream(stream, stream_arg, file_object):
225 while True:
212 while True:
226 line = (await stream.readline()).decode("utf8")
213 line = (await stream.readline()).decode("utf8")
227 if not line:
214 if not line:
228 break
215 break
229 if stream_arg:
216 if stream_arg:
230 self.shell.user_ns[stream_arg] = line
217 self.shell.user_ns[stream_arg] = line
231 else:
218 else:
232 file_object.write(line)
219 file_object.write(line)
233 file_object.flush()
220 file_object.flush()
234
221
235 async def _stream_communicate(process, cell):
222 async def _stream_communicate(process, cell):
236 process.stdin.write(cell)
223 process.stdin.write(cell)
237 process.stdin.close()
224 process.stdin.close()
238 stdout_task = asyncio.create_task(
225 stdout_task = asyncio.create_task(
239 _handle_stream(process.stdout, args.out, sys.stdout)
226 _handle_stream(process.stdout, args.out, sys.stdout)
240 )
227 )
241 stderr_task = asyncio.create_task(
228 stderr_task = asyncio.create_task(
242 _handle_stream(process.stderr, args.err, sys.stderr)
229 _handle_stream(process.stderr, args.err, sys.stderr)
243 )
230 )
244 await asyncio.wait([stdout_task, stderr_task])
231 await asyncio.wait([stdout_task, stderr_task])
245 await process.wait()
232 await process.wait()
246
233
247 policy = asyncio.get_event_loop_policy()
248 if sys.platform.startswith("win") and not isinstance(
249 policy, asyncio.WindowsProactorEventLoopPolicy
250 ):
251 # _do not_ overwrite the current policy
252 policy = asyncio.WindowsProactorEventLoopPolicy()
253
254 try:
255 loop = policy.get_event_loop()
256 except RuntimeError:
257 # closed loop, make a new one
258 loop = policy.new_event_loop()
259 policy.set_event_loop(loop)
260 argv = arg_split(line, posix=not sys.platform.startswith("win"))
234 argv = arg_split(line, posix=not sys.platform.startswith("win"))
261 args, cmd = self.shebang.parser.parse_known_args(argv)
235 args, cmd = self.shebang.parser.parse_known_args(argv)
236
262 try:
237 try:
263 p = loop.run_until_complete(
238 p = in_thread(
264 asyncio.create_subprocess_exec(
239 asyncio.create_subprocess_exec(
265 *cmd,
240 *cmd,
266 stdout=asyncio.subprocess.PIPE,
241 stdout=asyncio.subprocess.PIPE,
267 stderr=asyncio.subprocess.PIPE,
242 stderr=asyncio.subprocess.PIPE,
268 stdin=asyncio.subprocess.PIPE,
243 stdin=asyncio.subprocess.PIPE,
269 )
244 )
270 )
245 )
271 except OSError as e:
246 except OSError as e:
272 if e.errno == errno.ENOENT:
247 if e.errno == errno.ENOENT:
273 print("Couldn't find program: %r" % cmd[0])
248 print("Couldn't find program: %r" % cmd[0])
274 return
249 return
275 else:
250 else:
276 raise
251 raise
277
252
278 if not cell.endswith('\n'):
253 if not cell.endswith('\n'):
279 cell += '\n'
254 cell += '\n'
280 cell = cell.encode('utf8', 'replace')
255 cell = cell.encode('utf8', 'replace')
281 if args.bg:
256 if args.bg:
282 self.bg_processes.append(p)
257 self.bg_processes.append(p)
283 self._gc_bg_processes()
258 self._gc_bg_processes()
284 to_close = []
259 to_close = []
285 if args.out:
260 if args.out:
286 self.shell.user_ns[args.out] = p.stdout
261 self.shell.user_ns[args.out] = _AsyncIOProxy(p.stdout, event_loop)
287 else:
262 else:
288 to_close.append(p.stdout)
263 to_close.append(p.stdout)
289 if args.err:
264 if args.err:
290 self.shell.user_ns[args.err] = p.stderr
265 self.shell.user_ns[args.err] = _AsyncIOProxy(p.stderr, event_loop)
291 else:
266 else:
292 to_close.append(p.stderr)
267 to_close.append(p.stderr)
293 self.job_manager.new(self._run_script, p, cell, to_close, daemon=True)
268 event_loop.call_soon_threadsafe(
269 lambda: asyncio.Task(self._run_script(p, cell, to_close))
270 )
294 if args.proc:
271 if args.proc:
295 self.shell.user_ns[args.proc] = p
272 proc_proxy = _AsyncIOProxy(p, event_loop)
273 proc_proxy.stdout = _AsyncIOProxy(p.stdout, event_loop)
274 proc_proxy.stderr = _AsyncIOProxy(p.stderr, event_loop)
275 self.shell.user_ns[args.proc] = proc_proxy
296 return
276 return
297
277
298 try:
278 try:
299 loop.run_until_complete(_stream_communicate(p, cell))
279 in_thread(_stream_communicate(p, cell))
300 except KeyboardInterrupt:
280 except KeyboardInterrupt:
301 try:
281 try:
302 p.send_signal(signal.SIGINT)
282 p.send_signal(signal.SIGINT)
303 time.sleep(0.1)
283 in_thread(asyncio.wait_for(p.wait(), timeout=0.1))
304 if p.returncode is not None:
284 if p.returncode is not None:
305 print("Process is interrupted.")
285 print("Process is interrupted.")
306 return
286 return
307 p.terminate()
287 p.terminate()
308 time.sleep(0.1)
288 in_thread(asyncio.wait_for(p.wait(), timeout=0.1))
309 if p.returncode is not None:
289 if p.returncode is not None:
310 print("Process is terminated.")
290 print("Process is terminated.")
311 return
291 return
312 p.kill()
292 p.kill()
313 print("Process is killed.")
293 print("Process is killed.")
314 except OSError:
294 except OSError:
315 pass
295 pass
316 except Exception as e:
296 except Exception as e:
317 print("Error while terminating subprocess (pid=%i): %s" % (p.pid, e))
297 print("Error while terminating subprocess (pid=%i): %s" % (p.pid, e))
318 return
298 return
319 if args.raise_error and p.returncode!=0:
299
300 if args.raise_error and p.returncode != 0:
320 # If we get here and p.returncode is still None, we must have
301 # If we get here and p.returncode is still None, we must have
321 # killed it but not yet seen its return code. We don't wait for it,
302 # killed it but not yet seen its return code. We don't wait for it,
322 # in case it's stuck in uninterruptible sleep. -9 = SIGKILL
303 # in case it's stuck in uninterruptible sleep. -9 = SIGKILL
323 rc = p.returncode or -9
304 rc = p.returncode or -9
324 raise CalledProcessError(rc, cell)
305 raise CalledProcessError(rc, cell)
325
306
326 shebang.__skip_doctest__ = os.name != "posix"
307 shebang.__skip_doctest__ = os.name != "posix"
327
308
328 def _run_script(self, p, cell, to_close):
309 async def _run_script(self, p, cell, to_close):
329 """callback for running the script in the background"""
310 """callback for running the script in the background"""
311
330 p.stdin.write(cell)
312 p.stdin.write(cell)
313 await p.stdin.drain()
331 p.stdin.close()
314 p.stdin.close()
315 await p.stdin.wait_closed()
316 await p.wait()
317 # asyncio read pipes have no close
318 # but we should drain the data anyway
332 for s in to_close:
319 for s in to_close:
333 s.close()
320 await s.read()
334 p.wait()
321 self._gc_bg_processes()
335
322
336 @line_magic("killbgscripts")
323 @line_magic("killbgscripts")
337 def killbgscripts(self, _nouse_=''):
324 def killbgscripts(self, _nouse_=''):
338 """Kill all BG processes started by %%script and its family."""
325 """Kill all BG processes started by %%script and its family."""
339 self.kill_bg_processes()
326 self.kill_bg_processes()
340 print("All background processes were killed.")
327 print("All background processes were killed.")
341
328
342 def kill_bg_processes(self):
329 def kill_bg_processes(self):
343 """Kill all BG processes which are still running."""
330 """Kill all BG processes which are still running."""
344 if not self.bg_processes:
331 if not self.bg_processes:
345 return
332 return
346 for p in self.bg_processes:
333 for p in self.bg_processes:
347 if p.returncode is None:
334 if p.returncode is None:
348 try:
335 try:
349 p.send_signal(signal.SIGINT)
336 p.send_signal(signal.SIGINT)
350 except:
337 except:
351 pass
338 pass
352 time.sleep(0.1)
339 time.sleep(0.1)
353 self._gc_bg_processes()
340 self._gc_bg_processes()
354 if not self.bg_processes:
341 if not self.bg_processes:
355 return
342 return
356 for p in self.bg_processes:
343 for p in self.bg_processes:
357 if p.returncode is None:
344 if p.returncode is None:
358 try:
345 try:
359 p.terminate()
346 p.terminate()
360 except:
347 except:
361 pass
348 pass
362 time.sleep(0.1)
349 time.sleep(0.1)
363 self._gc_bg_processes()
350 self._gc_bg_processes()
364 if not self.bg_processes:
351 if not self.bg_processes:
365 return
352 return
366 for p in self.bg_processes:
353 for p in self.bg_processes:
367 if p.returncode is None:
354 if p.returncode is None:
368 try:
355 try:
369 p.kill()
356 p.kill()
370 except:
357 except:
371 pass
358 pass
372 self._gc_bg_processes()
359 self._gc_bg_processes()
373
360
374 def _gc_bg_processes(self):
361 def _gc_bg_processes(self):
375 self.bg_processes = [p for p in self.bg_processes if p.returncode is None]
362 self.bg_processes = [p for p in self.bg_processes if p.returncode is None]
@@ -1,1098 +1,1098 b''
1 # -*- coding: utf-8 -*-
1 # -*- coding: utf-8 -*-
2 """Tests for the key interactiveshell module.
2 """Tests for the key interactiveshell module.
3
3
4 Historically the main classes in interactiveshell have been under-tested. This
4 Historically the main classes in interactiveshell have been under-tested. This
5 module should grow as many single-method tests as possible to trap many of the
5 module should grow as many single-method tests as possible to trap many of the
6 recurring bugs we seem to encounter with high-level interaction.
6 recurring bugs we seem to encounter with high-level interaction.
7 """
7 """
8
8
9 # Copyright (c) IPython Development Team.
9 # Copyright (c) IPython Development Team.
10 # Distributed under the terms of the Modified BSD License.
10 # Distributed under the terms of the Modified BSD License.
11
11
12 import asyncio
12 import asyncio
13 import ast
13 import ast
14 import os
14 import os
15 import signal
15 import signal
16 import shutil
16 import shutil
17 import sys
17 import sys
18 import tempfile
18 import tempfile
19 import unittest
19 import unittest
20 from unittest import mock
20 from unittest import mock
21
21
22 from os.path import join
22 from os.path import join
23
23
24 from IPython.core.error import InputRejected
24 from IPython.core.error import InputRejected
25 from IPython.core.inputtransformer import InputTransformer
25 from IPython.core.inputtransformer import InputTransformer
26 from IPython.core import interactiveshell
26 from IPython.core import interactiveshell
27 from IPython.testing.decorators import (
27 from IPython.testing.decorators import (
28 skipif, skip_win32, onlyif_unicode_paths, onlyif_cmds_exist,
28 skipif, skip_win32, onlyif_unicode_paths, onlyif_cmds_exist,
29 )
29 )
30 from IPython.testing import tools as tt
30 from IPython.testing import tools as tt
31 from IPython.utils.process import find_cmd
31 from IPython.utils.process import find_cmd
32
32
33 #-----------------------------------------------------------------------------
33 #-----------------------------------------------------------------------------
34 # Globals
34 # Globals
35 #-----------------------------------------------------------------------------
35 #-----------------------------------------------------------------------------
36 # This is used by every single test, no point repeating it ad nauseam
36 # This is used by every single test, no point repeating it ad nauseam
37
37
38 #-----------------------------------------------------------------------------
38 #-----------------------------------------------------------------------------
39 # Tests
39 # Tests
40 #-----------------------------------------------------------------------------
40 #-----------------------------------------------------------------------------
41
41
42 class DerivedInterrupt(KeyboardInterrupt):
42 class DerivedInterrupt(KeyboardInterrupt):
43 pass
43 pass
44
44
45 class InteractiveShellTestCase(unittest.TestCase):
45 class InteractiveShellTestCase(unittest.TestCase):
46 def test_naked_string_cells(self):
46 def test_naked_string_cells(self):
47 """Test that cells with only naked strings are fully executed"""
47 """Test that cells with only naked strings are fully executed"""
48 # First, single-line inputs
48 # First, single-line inputs
49 ip.run_cell('"a"\n')
49 ip.run_cell('"a"\n')
50 self.assertEqual(ip.user_ns['_'], 'a')
50 self.assertEqual(ip.user_ns['_'], 'a')
51 # And also multi-line cells
51 # And also multi-line cells
52 ip.run_cell('"""a\nb"""\n')
52 ip.run_cell('"""a\nb"""\n')
53 self.assertEqual(ip.user_ns['_'], 'a\nb')
53 self.assertEqual(ip.user_ns['_'], 'a\nb')
54
54
55 def test_run_empty_cell(self):
55 def test_run_empty_cell(self):
56 """Just make sure we don't get a horrible error with a blank
56 """Just make sure we don't get a horrible error with a blank
57 cell of input. Yes, I did overlook that."""
57 cell of input. Yes, I did overlook that."""
58 old_xc = ip.execution_count
58 old_xc = ip.execution_count
59 res = ip.run_cell('')
59 res = ip.run_cell('')
60 self.assertEqual(ip.execution_count, old_xc)
60 self.assertEqual(ip.execution_count, old_xc)
61 self.assertEqual(res.execution_count, None)
61 self.assertEqual(res.execution_count, None)
62
62
63 def test_run_cell_multiline(self):
63 def test_run_cell_multiline(self):
64 """Multi-block, multi-line cells must execute correctly.
64 """Multi-block, multi-line cells must execute correctly.
65 """
65 """
66 src = '\n'.join(["x=1",
66 src = '\n'.join(["x=1",
67 "y=2",
67 "y=2",
68 "if 1:",
68 "if 1:",
69 " x += 1",
69 " x += 1",
70 " y += 1",])
70 " y += 1",])
71 res = ip.run_cell(src)
71 res = ip.run_cell(src)
72 self.assertEqual(ip.user_ns['x'], 2)
72 self.assertEqual(ip.user_ns['x'], 2)
73 self.assertEqual(ip.user_ns['y'], 3)
73 self.assertEqual(ip.user_ns['y'], 3)
74 self.assertEqual(res.success, True)
74 self.assertEqual(res.success, True)
75 self.assertEqual(res.result, None)
75 self.assertEqual(res.result, None)
76
76
77 def test_multiline_string_cells(self):
77 def test_multiline_string_cells(self):
78 "Code sprinkled with multiline strings should execute (GH-306)"
78 "Code sprinkled with multiline strings should execute (GH-306)"
79 ip.run_cell('tmp=0')
79 ip.run_cell('tmp=0')
80 self.assertEqual(ip.user_ns['tmp'], 0)
80 self.assertEqual(ip.user_ns['tmp'], 0)
81 res = ip.run_cell('tmp=1;"""a\nb"""\n')
81 res = ip.run_cell('tmp=1;"""a\nb"""\n')
82 self.assertEqual(ip.user_ns['tmp'], 1)
82 self.assertEqual(ip.user_ns['tmp'], 1)
83 self.assertEqual(res.success, True)
83 self.assertEqual(res.success, True)
84 self.assertEqual(res.result, "a\nb")
84 self.assertEqual(res.result, "a\nb")
85
85
86 def test_dont_cache_with_semicolon(self):
86 def test_dont_cache_with_semicolon(self):
87 "Ending a line with semicolon should not cache the returned object (GH-307)"
87 "Ending a line with semicolon should not cache the returned object (GH-307)"
88 oldlen = len(ip.user_ns['Out'])
88 oldlen = len(ip.user_ns['Out'])
89 for cell in ['1;', '1;1;']:
89 for cell in ['1;', '1;1;']:
90 res = ip.run_cell(cell, store_history=True)
90 res = ip.run_cell(cell, store_history=True)
91 newlen = len(ip.user_ns['Out'])
91 newlen = len(ip.user_ns['Out'])
92 self.assertEqual(oldlen, newlen)
92 self.assertEqual(oldlen, newlen)
93 self.assertIsNone(res.result)
93 self.assertIsNone(res.result)
94 i = 0
94 i = 0
95 #also test the default caching behavior
95 #also test the default caching behavior
96 for cell in ['1', '1;1']:
96 for cell in ['1', '1;1']:
97 ip.run_cell(cell, store_history=True)
97 ip.run_cell(cell, store_history=True)
98 newlen = len(ip.user_ns['Out'])
98 newlen = len(ip.user_ns['Out'])
99 i += 1
99 i += 1
100 self.assertEqual(oldlen+i, newlen)
100 self.assertEqual(oldlen+i, newlen)
101
101
102 def test_syntax_error(self):
102 def test_syntax_error(self):
103 res = ip.run_cell("raise = 3")
103 res = ip.run_cell("raise = 3")
104 self.assertIsInstance(res.error_before_exec, SyntaxError)
104 self.assertIsInstance(res.error_before_exec, SyntaxError)
105
105
106 def test_In_variable(self):
106 def test_In_variable(self):
107 "Verify that In variable grows with user input (GH-284)"
107 "Verify that In variable grows with user input (GH-284)"
108 oldlen = len(ip.user_ns['In'])
108 oldlen = len(ip.user_ns['In'])
109 ip.run_cell('1;', store_history=True)
109 ip.run_cell('1;', store_history=True)
110 newlen = len(ip.user_ns['In'])
110 newlen = len(ip.user_ns['In'])
111 self.assertEqual(oldlen+1, newlen)
111 self.assertEqual(oldlen+1, newlen)
112 self.assertEqual(ip.user_ns['In'][-1],'1;')
112 self.assertEqual(ip.user_ns['In'][-1],'1;')
113
113
114 def test_magic_names_in_string(self):
114 def test_magic_names_in_string(self):
115 ip.run_cell('a = """\n%exit\n"""')
115 ip.run_cell('a = """\n%exit\n"""')
116 self.assertEqual(ip.user_ns['a'], '\n%exit\n')
116 self.assertEqual(ip.user_ns['a'], '\n%exit\n')
117
117
118 def test_trailing_newline(self):
118 def test_trailing_newline(self):
119 """test that running !(command) does not raise a SyntaxError"""
119 """test that running !(command) does not raise a SyntaxError"""
120 ip.run_cell('!(true)\n', False)
120 ip.run_cell('!(true)\n', False)
121 ip.run_cell('!(true)\n\n\n', False)
121 ip.run_cell('!(true)\n\n\n', False)
122
122
123 def test_gh_597(self):
123 def test_gh_597(self):
124 """Pretty-printing lists of objects with non-ascii reprs may cause
124 """Pretty-printing lists of objects with non-ascii reprs may cause
125 problems."""
125 problems."""
126 class Spam(object):
126 class Spam(object):
127 def __repr__(self):
127 def __repr__(self):
128 return "\xe9"*50
128 return "\xe9"*50
129 import IPython.core.formatters
129 import IPython.core.formatters
130 f = IPython.core.formatters.PlainTextFormatter()
130 f = IPython.core.formatters.PlainTextFormatter()
131 f([Spam(),Spam()])
131 f([Spam(),Spam()])
132
132
133
133
134 def test_future_flags(self):
134 def test_future_flags(self):
135 """Check that future flags are used for parsing code (gh-777)"""
135 """Check that future flags are used for parsing code (gh-777)"""
136 ip.run_cell('from __future__ import barry_as_FLUFL')
136 ip.run_cell('from __future__ import barry_as_FLUFL')
137 try:
137 try:
138 ip.run_cell('prfunc_return_val = 1 <> 2')
138 ip.run_cell('prfunc_return_val = 1 <> 2')
139 assert 'prfunc_return_val' in ip.user_ns
139 assert 'prfunc_return_val' in ip.user_ns
140 finally:
140 finally:
141 # Reset compiler flags so we don't mess up other tests.
141 # Reset compiler flags so we don't mess up other tests.
142 ip.compile.reset_compiler_flags()
142 ip.compile.reset_compiler_flags()
143
143
144 def test_can_pickle(self):
144 def test_can_pickle(self):
145 "Can we pickle objects defined interactively (GH-29)"
145 "Can we pickle objects defined interactively (GH-29)"
146 ip = get_ipython()
146 ip = get_ipython()
147 ip.reset()
147 ip.reset()
148 ip.run_cell(("class Mylist(list):\n"
148 ip.run_cell(("class Mylist(list):\n"
149 " def __init__(self,x=[]):\n"
149 " def __init__(self,x=[]):\n"
150 " list.__init__(self,x)"))
150 " list.__init__(self,x)"))
151 ip.run_cell("w=Mylist([1,2,3])")
151 ip.run_cell("w=Mylist([1,2,3])")
152
152
153 from pickle import dumps
153 from pickle import dumps
154
154
155 # We need to swap in our main module - this is only necessary
155 # We need to swap in our main module - this is only necessary
156 # inside the test framework, because IPython puts the interactive module
156 # inside the test framework, because IPython puts the interactive module
157 # in place (but the test framework undoes this).
157 # in place (but the test framework undoes this).
158 _main = sys.modules['__main__']
158 _main = sys.modules['__main__']
159 sys.modules['__main__'] = ip.user_module
159 sys.modules['__main__'] = ip.user_module
160 try:
160 try:
161 res = dumps(ip.user_ns["w"])
161 res = dumps(ip.user_ns["w"])
162 finally:
162 finally:
163 sys.modules['__main__'] = _main
163 sys.modules['__main__'] = _main
164 self.assertTrue(isinstance(res, bytes))
164 self.assertTrue(isinstance(res, bytes))
165
165
166 def test_global_ns(self):
166 def test_global_ns(self):
167 "Code in functions must be able to access variables outside them."
167 "Code in functions must be able to access variables outside them."
168 ip = get_ipython()
168 ip = get_ipython()
169 ip.run_cell("a = 10")
169 ip.run_cell("a = 10")
170 ip.run_cell(("def f(x):\n"
170 ip.run_cell(("def f(x):\n"
171 " return x + a"))
171 " return x + a"))
172 ip.run_cell("b = f(12)")
172 ip.run_cell("b = f(12)")
173 self.assertEqual(ip.user_ns["b"], 22)
173 self.assertEqual(ip.user_ns["b"], 22)
174
174
175 def test_bad_custom_tb(self):
175 def test_bad_custom_tb(self):
176 """Check that InteractiveShell is protected from bad custom exception handlers"""
176 """Check that InteractiveShell is protected from bad custom exception handlers"""
177 ip.set_custom_exc((IOError,), lambda etype,value,tb: 1/0)
177 ip.set_custom_exc((IOError,), lambda etype,value,tb: 1/0)
178 self.assertEqual(ip.custom_exceptions, (IOError,))
178 self.assertEqual(ip.custom_exceptions, (IOError,))
179 with tt.AssertPrints("Custom TB Handler failed", channel='stderr'):
179 with tt.AssertPrints("Custom TB Handler failed", channel='stderr'):
180 ip.run_cell(u'raise IOError("foo")')
180 ip.run_cell(u'raise IOError("foo")')
181 self.assertEqual(ip.custom_exceptions, ())
181 self.assertEqual(ip.custom_exceptions, ())
182
182
183 def test_bad_custom_tb_return(self):
183 def test_bad_custom_tb_return(self):
184 """Check that InteractiveShell is protected from bad return types in custom exception handlers"""
184 """Check that InteractiveShell is protected from bad return types in custom exception handlers"""
185 ip.set_custom_exc((NameError,),lambda etype,value,tb, tb_offset=None: 1)
185 ip.set_custom_exc((NameError,),lambda etype,value,tb, tb_offset=None: 1)
186 self.assertEqual(ip.custom_exceptions, (NameError,))
186 self.assertEqual(ip.custom_exceptions, (NameError,))
187 with tt.AssertPrints("Custom TB Handler failed", channel='stderr'):
187 with tt.AssertPrints("Custom TB Handler failed", channel='stderr'):
188 ip.run_cell(u'a=abracadabra')
188 ip.run_cell(u'a=abracadabra')
189 self.assertEqual(ip.custom_exceptions, ())
189 self.assertEqual(ip.custom_exceptions, ())
190
190
191 def test_drop_by_id(self):
191 def test_drop_by_id(self):
192 myvars = {"a":object(), "b":object(), "c": object()}
192 myvars = {"a":object(), "b":object(), "c": object()}
193 ip.push(myvars, interactive=False)
193 ip.push(myvars, interactive=False)
194 for name in myvars:
194 for name in myvars:
195 assert name in ip.user_ns, name
195 assert name in ip.user_ns, name
196 assert name in ip.user_ns_hidden, name
196 assert name in ip.user_ns_hidden, name
197 ip.user_ns['b'] = 12
197 ip.user_ns['b'] = 12
198 ip.drop_by_id(myvars)
198 ip.drop_by_id(myvars)
199 for name in ["a", "c"]:
199 for name in ["a", "c"]:
200 assert name not in ip.user_ns, name
200 assert name not in ip.user_ns, name
201 assert name not in ip.user_ns_hidden, name
201 assert name not in ip.user_ns_hidden, name
202 assert ip.user_ns['b'] == 12
202 assert ip.user_ns['b'] == 12
203 ip.reset()
203 ip.reset()
204
204
205 def test_var_expand(self):
205 def test_var_expand(self):
206 ip.user_ns['f'] = u'Ca\xf1o'
206 ip.user_ns['f'] = u'Ca\xf1o'
207 self.assertEqual(ip.var_expand(u'echo $f'), u'echo Ca\xf1o')
207 self.assertEqual(ip.var_expand(u'echo $f'), u'echo Ca\xf1o')
208 self.assertEqual(ip.var_expand(u'echo {f}'), u'echo Ca\xf1o')
208 self.assertEqual(ip.var_expand(u'echo {f}'), u'echo Ca\xf1o')
209 self.assertEqual(ip.var_expand(u'echo {f[:-1]}'), u'echo Ca\xf1')
209 self.assertEqual(ip.var_expand(u'echo {f[:-1]}'), u'echo Ca\xf1')
210 self.assertEqual(ip.var_expand(u'echo {1*2}'), u'echo 2')
210 self.assertEqual(ip.var_expand(u'echo {1*2}'), u'echo 2')
211
211
212 self.assertEqual(ip.var_expand(u"grep x | awk '{print $1}'"), u"grep x | awk '{print $1}'")
212 self.assertEqual(ip.var_expand(u"grep x | awk '{print $1}'"), u"grep x | awk '{print $1}'")
213
213
214 ip.user_ns['f'] = b'Ca\xc3\xb1o'
214 ip.user_ns['f'] = b'Ca\xc3\xb1o'
215 # This should not raise any exception:
215 # This should not raise any exception:
216 ip.var_expand(u'echo $f')
216 ip.var_expand(u'echo $f')
217
217
218 def test_var_expand_local(self):
218 def test_var_expand_local(self):
219 """Test local variable expansion in !system and %magic calls"""
219 """Test local variable expansion in !system and %magic calls"""
220 # !system
220 # !system
221 ip.run_cell(
221 ip.run_cell(
222 "def test():\n"
222 "def test():\n"
223 ' lvar = "ttt"\n'
223 ' lvar = "ttt"\n'
224 " ret = !echo {lvar}\n"
224 " ret = !echo {lvar}\n"
225 " return ret[0]\n"
225 " return ret[0]\n"
226 )
226 )
227 res = ip.user_ns["test"]()
227 res = ip.user_ns["test"]()
228 self.assertIn("ttt", res)
228 self.assertIn("ttt", res)
229
229
230 # %magic
230 # %magic
231 ip.run_cell(
231 ip.run_cell(
232 "def makemacro():\n"
232 "def makemacro():\n"
233 ' macroname = "macro_var_expand_locals"\n'
233 ' macroname = "macro_var_expand_locals"\n'
234 " %macro {macroname} codestr\n"
234 " %macro {macroname} codestr\n"
235 )
235 )
236 ip.user_ns["codestr"] = "str(12)"
236 ip.user_ns["codestr"] = "str(12)"
237 ip.run_cell("makemacro()")
237 ip.run_cell("makemacro()")
238 self.assertIn("macro_var_expand_locals", ip.user_ns)
238 self.assertIn("macro_var_expand_locals", ip.user_ns)
239
239
240 def test_var_expand_self(self):
240 def test_var_expand_self(self):
241 """Test variable expansion with the name 'self', which was failing.
241 """Test variable expansion with the name 'self', which was failing.
242
242
243 See https://github.com/ipython/ipython/issues/1878#issuecomment-7698218
243 See https://github.com/ipython/ipython/issues/1878#issuecomment-7698218
244 """
244 """
245 ip.run_cell(
245 ip.run_cell(
246 "class cTest:\n"
246 "class cTest:\n"
247 ' classvar="see me"\n'
247 ' classvar="see me"\n'
248 " def test(self):\n"
248 " def test(self):\n"
249 " res = !echo Variable: {self.classvar}\n"
249 " res = !echo Variable: {self.classvar}\n"
250 " return res[0]\n"
250 " return res[0]\n"
251 )
251 )
252 self.assertIn("see me", ip.user_ns["cTest"]().test())
252 self.assertIn("see me", ip.user_ns["cTest"]().test())
253
253
254 def test_bad_var_expand(self):
254 def test_bad_var_expand(self):
255 """var_expand on invalid formats shouldn't raise"""
255 """var_expand on invalid formats shouldn't raise"""
256 # SyntaxError
256 # SyntaxError
257 self.assertEqual(ip.var_expand(u"{'a':5}"), u"{'a':5}")
257 self.assertEqual(ip.var_expand(u"{'a':5}"), u"{'a':5}")
258 # NameError
258 # NameError
259 self.assertEqual(ip.var_expand(u"{asdf}"), u"{asdf}")
259 self.assertEqual(ip.var_expand(u"{asdf}"), u"{asdf}")
260 # ZeroDivisionError
260 # ZeroDivisionError
261 self.assertEqual(ip.var_expand(u"{1/0}"), u"{1/0}")
261 self.assertEqual(ip.var_expand(u"{1/0}"), u"{1/0}")
262
262
263 def test_silent_postexec(self):
263 def test_silent_postexec(self):
264 """run_cell(silent=True) doesn't invoke pre/post_run_cell callbacks"""
264 """run_cell(silent=True) doesn't invoke pre/post_run_cell callbacks"""
265 pre_explicit = mock.Mock()
265 pre_explicit = mock.Mock()
266 pre_always = mock.Mock()
266 pre_always = mock.Mock()
267 post_explicit = mock.Mock()
267 post_explicit = mock.Mock()
268 post_always = mock.Mock()
268 post_always = mock.Mock()
269 all_mocks = [pre_explicit, pre_always, post_explicit, post_always]
269 all_mocks = [pre_explicit, pre_always, post_explicit, post_always]
270
270
271 ip.events.register('pre_run_cell', pre_explicit)
271 ip.events.register('pre_run_cell', pre_explicit)
272 ip.events.register('pre_execute', pre_always)
272 ip.events.register('pre_execute', pre_always)
273 ip.events.register('post_run_cell', post_explicit)
273 ip.events.register('post_run_cell', post_explicit)
274 ip.events.register('post_execute', post_always)
274 ip.events.register('post_execute', post_always)
275
275
276 try:
276 try:
277 ip.run_cell("1", silent=True)
277 ip.run_cell("1", silent=True)
278 assert pre_always.called
278 assert pre_always.called
279 assert not pre_explicit.called
279 assert not pre_explicit.called
280 assert post_always.called
280 assert post_always.called
281 assert not post_explicit.called
281 assert not post_explicit.called
282 # double-check that non-silent exec did what we expected
282 # double-check that non-silent exec did what we expected
283 # silent to avoid
283 # silent to avoid
284 ip.run_cell("1")
284 ip.run_cell("1")
285 assert pre_explicit.called
285 assert pre_explicit.called
286 assert post_explicit.called
286 assert post_explicit.called
287 info, = pre_explicit.call_args[0]
287 info, = pre_explicit.call_args[0]
288 result, = post_explicit.call_args[0]
288 result, = post_explicit.call_args[0]
289 self.assertEqual(info, result.info)
289 self.assertEqual(info, result.info)
290 # check that post hooks are always called
290 # check that post hooks are always called
291 [m.reset_mock() for m in all_mocks]
291 [m.reset_mock() for m in all_mocks]
292 ip.run_cell("syntax error")
292 ip.run_cell("syntax error")
293 assert pre_always.called
293 assert pre_always.called
294 assert pre_explicit.called
294 assert pre_explicit.called
295 assert post_always.called
295 assert post_always.called
296 assert post_explicit.called
296 assert post_explicit.called
297 info, = pre_explicit.call_args[0]
297 info, = pre_explicit.call_args[0]
298 result, = post_explicit.call_args[0]
298 result, = post_explicit.call_args[0]
299 self.assertEqual(info, result.info)
299 self.assertEqual(info, result.info)
300 finally:
300 finally:
301 # remove post-exec
301 # remove post-exec
302 ip.events.unregister('pre_run_cell', pre_explicit)
302 ip.events.unregister('pre_run_cell', pre_explicit)
303 ip.events.unregister('pre_execute', pre_always)
303 ip.events.unregister('pre_execute', pre_always)
304 ip.events.unregister('post_run_cell', post_explicit)
304 ip.events.unregister('post_run_cell', post_explicit)
305 ip.events.unregister('post_execute', post_always)
305 ip.events.unregister('post_execute', post_always)
306
306
307 def test_silent_noadvance(self):
307 def test_silent_noadvance(self):
308 """run_cell(silent=True) doesn't advance execution_count"""
308 """run_cell(silent=True) doesn't advance execution_count"""
309 ec = ip.execution_count
309 ec = ip.execution_count
310 # silent should force store_history=False
310 # silent should force store_history=False
311 ip.run_cell("1", store_history=True, silent=True)
311 ip.run_cell("1", store_history=True, silent=True)
312
312
313 self.assertEqual(ec, ip.execution_count)
313 self.assertEqual(ec, ip.execution_count)
314 # double-check that non-silent exec did what we expected
314 # double-check that non-silent exec did what we expected
315 # silent to avoid
315 # silent to avoid
316 ip.run_cell("1", store_history=True)
316 ip.run_cell("1", store_history=True)
317 self.assertEqual(ec+1, ip.execution_count)
317 self.assertEqual(ec+1, ip.execution_count)
318
318
319 def test_silent_nodisplayhook(self):
319 def test_silent_nodisplayhook(self):
320 """run_cell(silent=True) doesn't trigger displayhook"""
320 """run_cell(silent=True) doesn't trigger displayhook"""
321 d = dict(called=False)
321 d = dict(called=False)
322
322
323 trap = ip.display_trap
323 trap = ip.display_trap
324 save_hook = trap.hook
324 save_hook = trap.hook
325
325
326 def failing_hook(*args, **kwargs):
326 def failing_hook(*args, **kwargs):
327 d['called'] = True
327 d['called'] = True
328
328
329 try:
329 try:
330 trap.hook = failing_hook
330 trap.hook = failing_hook
331 res = ip.run_cell("1", silent=True)
331 res = ip.run_cell("1", silent=True)
332 self.assertFalse(d['called'])
332 self.assertFalse(d['called'])
333 self.assertIsNone(res.result)
333 self.assertIsNone(res.result)
334 # double-check that non-silent exec did what we expected
334 # double-check that non-silent exec did what we expected
335 # silent to avoid
335 # silent to avoid
336 ip.run_cell("1")
336 ip.run_cell("1")
337 self.assertTrue(d['called'])
337 self.assertTrue(d['called'])
338 finally:
338 finally:
339 trap.hook = save_hook
339 trap.hook = save_hook
340
340
341 def test_ofind_line_magic(self):
341 def test_ofind_line_magic(self):
342 from IPython.core.magic import register_line_magic
342 from IPython.core.magic import register_line_magic
343
343
344 @register_line_magic
344 @register_line_magic
345 def lmagic(line):
345 def lmagic(line):
346 "A line magic"
346 "A line magic"
347
347
348 # Get info on line magic
348 # Get info on line magic
349 lfind = ip._ofind("lmagic")
349 lfind = ip._ofind("lmagic")
350 info = dict(
350 info = dict(
351 found=True,
351 found=True,
352 isalias=False,
352 isalias=False,
353 ismagic=True,
353 ismagic=True,
354 namespace="IPython internal",
354 namespace="IPython internal",
355 obj=lmagic,
355 obj=lmagic,
356 parent=None,
356 parent=None,
357 )
357 )
358 self.assertEqual(lfind, info)
358 self.assertEqual(lfind, info)
359
359
360 def test_ofind_cell_magic(self):
360 def test_ofind_cell_magic(self):
361 from IPython.core.magic import register_cell_magic
361 from IPython.core.magic import register_cell_magic
362
362
363 @register_cell_magic
363 @register_cell_magic
364 def cmagic(line, cell):
364 def cmagic(line, cell):
365 "A cell magic"
365 "A cell magic"
366
366
367 # Get info on cell magic
367 # Get info on cell magic
368 find = ip._ofind("cmagic")
368 find = ip._ofind("cmagic")
369 info = dict(
369 info = dict(
370 found=True,
370 found=True,
371 isalias=False,
371 isalias=False,
372 ismagic=True,
372 ismagic=True,
373 namespace="IPython internal",
373 namespace="IPython internal",
374 obj=cmagic,
374 obj=cmagic,
375 parent=None,
375 parent=None,
376 )
376 )
377 self.assertEqual(find, info)
377 self.assertEqual(find, info)
378
378
379 def test_ofind_property_with_error(self):
379 def test_ofind_property_with_error(self):
380 class A(object):
380 class A(object):
381 @property
381 @property
382 def foo(self):
382 def foo(self):
383 raise NotImplementedError()
383 raise NotImplementedError()
384 a = A()
384 a = A()
385
385
386 found = ip._ofind('a.foo', [('locals', locals())])
386 found = ip._ofind('a.foo', [('locals', locals())])
387 info = dict(found=True, isalias=False, ismagic=False,
387 info = dict(found=True, isalias=False, ismagic=False,
388 namespace='locals', obj=A.foo, parent=a)
388 namespace='locals', obj=A.foo, parent=a)
389 self.assertEqual(found, info)
389 self.assertEqual(found, info)
390
390
391 def test_ofind_multiple_attribute_lookups(self):
391 def test_ofind_multiple_attribute_lookups(self):
392 class A(object):
392 class A(object):
393 @property
393 @property
394 def foo(self):
394 def foo(self):
395 raise NotImplementedError()
395 raise NotImplementedError()
396
396
397 a = A()
397 a = A()
398 a.a = A()
398 a.a = A()
399 a.a.a = A()
399 a.a.a = A()
400
400
401 found = ip._ofind('a.a.a.foo', [('locals', locals())])
401 found = ip._ofind('a.a.a.foo', [('locals', locals())])
402 info = dict(found=True, isalias=False, ismagic=False,
402 info = dict(found=True, isalias=False, ismagic=False,
403 namespace='locals', obj=A.foo, parent=a.a.a)
403 namespace='locals', obj=A.foo, parent=a.a.a)
404 self.assertEqual(found, info)
404 self.assertEqual(found, info)
405
405
406 def test_ofind_slotted_attributes(self):
406 def test_ofind_slotted_attributes(self):
407 class A(object):
407 class A(object):
408 __slots__ = ['foo']
408 __slots__ = ['foo']
409 def __init__(self):
409 def __init__(self):
410 self.foo = 'bar'
410 self.foo = 'bar'
411
411
412 a = A()
412 a = A()
413 found = ip._ofind('a.foo', [('locals', locals())])
413 found = ip._ofind('a.foo', [('locals', locals())])
414 info = dict(found=True, isalias=False, ismagic=False,
414 info = dict(found=True, isalias=False, ismagic=False,
415 namespace='locals', obj=a.foo, parent=a)
415 namespace='locals', obj=a.foo, parent=a)
416 self.assertEqual(found, info)
416 self.assertEqual(found, info)
417
417
418 found = ip._ofind('a.bar', [('locals', locals())])
418 found = ip._ofind('a.bar', [('locals', locals())])
419 info = dict(found=False, isalias=False, ismagic=False,
419 info = dict(found=False, isalias=False, ismagic=False,
420 namespace=None, obj=None, parent=a)
420 namespace=None, obj=None, parent=a)
421 self.assertEqual(found, info)
421 self.assertEqual(found, info)
422
422
423 def test_ofind_prefers_property_to_instance_level_attribute(self):
423 def test_ofind_prefers_property_to_instance_level_attribute(self):
424 class A(object):
424 class A(object):
425 @property
425 @property
426 def foo(self):
426 def foo(self):
427 return 'bar'
427 return 'bar'
428 a = A()
428 a = A()
429 a.__dict__["foo"] = "baz"
429 a.__dict__["foo"] = "baz"
430 self.assertEqual(a.foo, "bar")
430 self.assertEqual(a.foo, "bar")
431 found = ip._ofind("a.foo", [("locals", locals())])
431 found = ip._ofind("a.foo", [("locals", locals())])
432 self.assertIs(found["obj"], A.foo)
432 self.assertIs(found["obj"], A.foo)
433
433
434 def test_custom_syntaxerror_exception(self):
434 def test_custom_syntaxerror_exception(self):
435 called = []
435 called = []
436 def my_handler(shell, etype, value, tb, tb_offset=None):
436 def my_handler(shell, etype, value, tb, tb_offset=None):
437 called.append(etype)
437 called.append(etype)
438 shell.showtraceback((etype, value, tb), tb_offset=tb_offset)
438 shell.showtraceback((etype, value, tb), tb_offset=tb_offset)
439
439
440 ip.set_custom_exc((SyntaxError,), my_handler)
440 ip.set_custom_exc((SyntaxError,), my_handler)
441 try:
441 try:
442 ip.run_cell("1f")
442 ip.run_cell("1f")
443 # Check that this was called, and only once.
443 # Check that this was called, and only once.
444 self.assertEqual(called, [SyntaxError])
444 self.assertEqual(called, [SyntaxError])
445 finally:
445 finally:
446 # Reset the custom exception hook
446 # Reset the custom exception hook
447 ip.set_custom_exc((), None)
447 ip.set_custom_exc((), None)
448
448
449 def test_custom_exception(self):
449 def test_custom_exception(self):
450 called = []
450 called = []
451 def my_handler(shell, etype, value, tb, tb_offset=None):
451 def my_handler(shell, etype, value, tb, tb_offset=None):
452 called.append(etype)
452 called.append(etype)
453 shell.showtraceback((etype, value, tb), tb_offset=tb_offset)
453 shell.showtraceback((etype, value, tb), tb_offset=tb_offset)
454
454
455 ip.set_custom_exc((ValueError,), my_handler)
455 ip.set_custom_exc((ValueError,), my_handler)
456 try:
456 try:
457 res = ip.run_cell("raise ValueError('test')")
457 res = ip.run_cell("raise ValueError('test')")
458 # Check that this was called, and only once.
458 # Check that this was called, and only once.
459 self.assertEqual(called, [ValueError])
459 self.assertEqual(called, [ValueError])
460 # Check that the error is on the result object
460 # Check that the error is on the result object
461 self.assertIsInstance(res.error_in_exec, ValueError)
461 self.assertIsInstance(res.error_in_exec, ValueError)
462 finally:
462 finally:
463 # Reset the custom exception hook
463 # Reset the custom exception hook
464 ip.set_custom_exc((), None)
464 ip.set_custom_exc((), None)
465
465
466 @mock.patch("builtins.print")
466 @mock.patch("builtins.print")
467 def test_showtraceback_with_surrogates(self, mocked_print):
467 def test_showtraceback_with_surrogates(self, mocked_print):
468 values = []
468 values = []
469
469
470 def mock_print_func(value, sep=" ", end="\n", file=sys.stdout, flush=False):
470 def mock_print_func(value, sep=" ", end="\n", file=sys.stdout, flush=False):
471 values.append(value)
471 values.append(value)
472 if value == chr(0xD8FF):
472 if value == chr(0xD8FF):
473 raise UnicodeEncodeError("utf-8", chr(0xD8FF), 0, 1, "")
473 raise UnicodeEncodeError("utf-8", chr(0xD8FF), 0, 1, "")
474
474
475 # mock builtins.print
475 # mock builtins.print
476 mocked_print.side_effect = mock_print_func
476 mocked_print.side_effect = mock_print_func
477
477
478 # ip._showtraceback() is replaced in globalipapp.py.
478 # ip._showtraceback() is replaced in globalipapp.py.
479 # Call original method to test.
479 # Call original method to test.
480 interactiveshell.InteractiveShell._showtraceback(ip, None, None, chr(0xD8FF))
480 interactiveshell.InteractiveShell._showtraceback(ip, None, None, chr(0xD8FF))
481
481
482 self.assertEqual(mocked_print.call_count, 2)
482 self.assertEqual(mocked_print.call_count, 2)
483 self.assertEqual(values, [chr(0xD8FF), "\\ud8ff"])
483 self.assertEqual(values, [chr(0xD8FF), "\\ud8ff"])
484
484
485 def test_mktempfile(self):
485 def test_mktempfile(self):
486 filename = ip.mktempfile()
486 filename = ip.mktempfile()
487 # Check that we can open the file again on Windows
487 # Check that we can open the file again on Windows
488 with open(filename, 'w') as f:
488 with open(filename, 'w') as f:
489 f.write('abc')
489 f.write('abc')
490
490
491 filename = ip.mktempfile(data='blah')
491 filename = ip.mktempfile(data='blah')
492 with open(filename, 'r') as f:
492 with open(filename, 'r') as f:
493 self.assertEqual(f.read(), 'blah')
493 self.assertEqual(f.read(), 'blah')
494
494
495 def test_new_main_mod(self):
495 def test_new_main_mod(self):
496 # Smoketest to check that this accepts a unicode module name
496 # Smoketest to check that this accepts a unicode module name
497 name = u'jiefmw'
497 name = u'jiefmw'
498 mod = ip.new_main_mod(u'%s.py' % name, name)
498 mod = ip.new_main_mod(u'%s.py' % name, name)
499 self.assertEqual(mod.__name__, name)
499 self.assertEqual(mod.__name__, name)
500
500
501 def test_get_exception_only(self):
501 def test_get_exception_only(self):
502 try:
502 try:
503 raise KeyboardInterrupt
503 raise KeyboardInterrupt
504 except KeyboardInterrupt:
504 except KeyboardInterrupt:
505 msg = ip.get_exception_only()
505 msg = ip.get_exception_only()
506 self.assertEqual(msg, 'KeyboardInterrupt\n')
506 self.assertEqual(msg, 'KeyboardInterrupt\n')
507
507
508 try:
508 try:
509 raise DerivedInterrupt("foo")
509 raise DerivedInterrupt("foo")
510 except KeyboardInterrupt:
510 except KeyboardInterrupt:
511 msg = ip.get_exception_only()
511 msg = ip.get_exception_only()
512 self.assertEqual(msg, 'IPython.core.tests.test_interactiveshell.DerivedInterrupt: foo\n')
512 self.assertEqual(msg, 'IPython.core.tests.test_interactiveshell.DerivedInterrupt: foo\n')
513
513
514 def test_inspect_text(self):
514 def test_inspect_text(self):
515 ip.run_cell('a = 5')
515 ip.run_cell('a = 5')
516 text = ip.object_inspect_text('a')
516 text = ip.object_inspect_text('a')
517 self.assertIsInstance(text, str)
517 self.assertIsInstance(text, str)
518
518
519 def test_last_execution_result(self):
519 def test_last_execution_result(self):
520 """ Check that last execution result gets set correctly (GH-10702) """
520 """ Check that last execution result gets set correctly (GH-10702) """
521 result = ip.run_cell('a = 5; a')
521 result = ip.run_cell('a = 5; a')
522 self.assertTrue(ip.last_execution_succeeded)
522 self.assertTrue(ip.last_execution_succeeded)
523 self.assertEqual(ip.last_execution_result.result, 5)
523 self.assertEqual(ip.last_execution_result.result, 5)
524
524
525 result = ip.run_cell('a = x_invalid_id_x')
525 result = ip.run_cell('a = x_invalid_id_x')
526 self.assertFalse(ip.last_execution_succeeded)
526 self.assertFalse(ip.last_execution_succeeded)
527 self.assertFalse(ip.last_execution_result.success)
527 self.assertFalse(ip.last_execution_result.success)
528 self.assertIsInstance(ip.last_execution_result.error_in_exec, NameError)
528 self.assertIsInstance(ip.last_execution_result.error_in_exec, NameError)
529
529
530 def test_reset_aliasing(self):
530 def test_reset_aliasing(self):
531 """ Check that standard posix aliases work after %reset. """
531 """ Check that standard posix aliases work after %reset. """
532 if os.name != 'posix':
532 if os.name != 'posix':
533 return
533 return
534
534
535 ip.reset()
535 ip.reset()
536 for cmd in ('clear', 'more', 'less', 'man'):
536 for cmd in ('clear', 'more', 'less', 'man'):
537 res = ip.run_cell('%' + cmd)
537 res = ip.run_cell('%' + cmd)
538 self.assertEqual(res.success, True)
538 self.assertEqual(res.success, True)
539
539
540
540
541 class TestSafeExecfileNonAsciiPath(unittest.TestCase):
541 class TestSafeExecfileNonAsciiPath(unittest.TestCase):
542
542
543 @onlyif_unicode_paths
543 @onlyif_unicode_paths
544 def setUp(self):
544 def setUp(self):
545 self.BASETESTDIR = tempfile.mkdtemp()
545 self.BASETESTDIR = tempfile.mkdtemp()
546 self.TESTDIR = join(self.BASETESTDIR, u"Γ₯Àâ")
546 self.TESTDIR = join(self.BASETESTDIR, u"Γ₯Àâ")
547 os.mkdir(self.TESTDIR)
547 os.mkdir(self.TESTDIR)
548 with open(join(self.TESTDIR, u"Γ₯Àâtestscript.py"), "w") as sfile:
548 with open(join(self.TESTDIR, u"Γ₯Àâtestscript.py"), "w") as sfile:
549 sfile.write("pass\n")
549 sfile.write("pass\n")
550 self.oldpath = os.getcwd()
550 self.oldpath = os.getcwd()
551 os.chdir(self.TESTDIR)
551 os.chdir(self.TESTDIR)
552 self.fname = u"Γ₯Àâtestscript.py"
552 self.fname = u"Γ₯Àâtestscript.py"
553
553
554 def tearDown(self):
554 def tearDown(self):
555 os.chdir(self.oldpath)
555 os.chdir(self.oldpath)
556 shutil.rmtree(self.BASETESTDIR)
556 shutil.rmtree(self.BASETESTDIR)
557
557
558 @onlyif_unicode_paths
558 @onlyif_unicode_paths
559 def test_1(self):
559 def test_1(self):
560 """Test safe_execfile with non-ascii path
560 """Test safe_execfile with non-ascii path
561 """
561 """
562 ip.safe_execfile(self.fname, {}, raise_exceptions=True)
562 ip.safe_execfile(self.fname, {}, raise_exceptions=True)
563
563
564 class ExitCodeChecks(tt.TempFileMixin):
564 class ExitCodeChecks(tt.TempFileMixin):
565
565
566 def setUp(self):
566 def setUp(self):
567 self.system = ip.system_raw
567 self.system = ip.system_raw
568
568
569 def test_exit_code_ok(self):
569 def test_exit_code_ok(self):
570 self.system('exit 0')
570 self.system('exit 0')
571 self.assertEqual(ip.user_ns['_exit_code'], 0)
571 self.assertEqual(ip.user_ns['_exit_code'], 0)
572
572
573 def test_exit_code_error(self):
573 def test_exit_code_error(self):
574 self.system('exit 1')
574 self.system('exit 1')
575 self.assertEqual(ip.user_ns['_exit_code'], 1)
575 self.assertEqual(ip.user_ns['_exit_code'], 1)
576
576
577 @skipif(not hasattr(signal, 'SIGALRM'))
577 @skipif(not hasattr(signal, 'SIGALRM'))
578 def test_exit_code_signal(self):
578 def test_exit_code_signal(self):
579 self.mktmp("import signal, time\n"
579 self.mktmp("import signal, time\n"
580 "signal.setitimer(signal.ITIMER_REAL, 0.1)\n"
580 "signal.setitimer(signal.ITIMER_REAL, 0.1)\n"
581 "time.sleep(1)\n")
581 "time.sleep(1)\n")
582 self.system("%s %s" % (sys.executable, self.fname))
582 self.system("%s %s" % (sys.executable, self.fname))
583 self.assertEqual(ip.user_ns['_exit_code'], -signal.SIGALRM)
583 self.assertEqual(ip.user_ns['_exit_code'], -signal.SIGALRM)
584
584
585 @onlyif_cmds_exist("csh")
585 @onlyif_cmds_exist("csh")
586 def test_exit_code_signal_csh(self):
586 def test_exit_code_signal_csh(self):
587 SHELL = os.environ.get('SHELL', None)
587 SHELL = os.environ.get('SHELL', None)
588 os.environ['SHELL'] = find_cmd("csh")
588 os.environ['SHELL'] = find_cmd("csh")
589 try:
589 try:
590 self.test_exit_code_signal()
590 self.test_exit_code_signal()
591 finally:
591 finally:
592 if SHELL is not None:
592 if SHELL is not None:
593 os.environ['SHELL'] = SHELL
593 os.environ['SHELL'] = SHELL
594 else:
594 else:
595 del os.environ['SHELL']
595 del os.environ['SHELL']
596
596
597
597
598 class TestSystemRaw(ExitCodeChecks):
598 class TestSystemRaw(ExitCodeChecks):
599
599
600 def setUp(self):
600 def setUp(self):
601 super().setUp()
601 super().setUp()
602 self.system = ip.system_raw
602 self.system = ip.system_raw
603
603
604 @onlyif_unicode_paths
604 @onlyif_unicode_paths
605 def test_1(self):
605 def test_1(self):
606 """Test system_raw with non-ascii cmd
606 """Test system_raw with non-ascii cmd
607 """
607 """
608 cmd = u'''python -c "'Γ₯Àâ'" '''
608 cmd = u'''python -c "'Γ₯Àâ'" '''
609 ip.system_raw(cmd)
609 ip.system_raw(cmd)
610
610
611 @mock.patch('subprocess.call', side_effect=KeyboardInterrupt)
611 @mock.patch('subprocess.call', side_effect=KeyboardInterrupt)
612 @mock.patch('os.system', side_effect=KeyboardInterrupt)
612 @mock.patch('os.system', side_effect=KeyboardInterrupt)
613 def test_control_c(self, *mocks):
613 def test_control_c(self, *mocks):
614 try:
614 try:
615 self.system("sleep 1 # wont happen")
615 self.system("sleep 1 # wont happen")
616 except KeyboardInterrupt:
616 except KeyboardInterrupt:
617 self.fail(
617 self.fail(
618 "system call should intercept "
618 "system call should intercept "
619 "keyboard interrupt from subprocess.call"
619 "keyboard interrupt from subprocess.call"
620 )
620 )
621 self.assertEqual(ip.user_ns["_exit_code"], -signal.SIGINT)
621 self.assertEqual(ip.user_ns["_exit_code"], -signal.SIGINT)
622
622
623 def test_magic_warnings(self):
623 def test_magic_warnings(self):
624 for magic_cmd in ("ls", "pip", "conda", "cd"):
624 for magic_cmd in ("ls", "pip", "conda", "cd"):
625 with self.assertWarnsRegex(Warning, "You executed the system command"):
625 with self.assertWarnsRegex(Warning, "You executed the system command"):
626 ip.system_raw(magic_cmd)
626 ip.system_raw(magic_cmd)
627
627
628 # TODO: Exit codes are currently ignored on Windows.
628 # TODO: Exit codes are currently ignored on Windows.
629 class TestSystemPipedExitCode(ExitCodeChecks):
629 class TestSystemPipedExitCode(ExitCodeChecks):
630
630
631 def setUp(self):
631 def setUp(self):
632 super().setUp()
632 super().setUp()
633 self.system = ip.system_piped
633 self.system = ip.system_piped
634
634
635 @skip_win32
635 @skip_win32
636 def test_exit_code_ok(self):
636 def test_exit_code_ok(self):
637 ExitCodeChecks.test_exit_code_ok(self)
637 ExitCodeChecks.test_exit_code_ok(self)
638
638
639 @skip_win32
639 @skip_win32
640 def test_exit_code_error(self):
640 def test_exit_code_error(self):
641 ExitCodeChecks.test_exit_code_error(self)
641 ExitCodeChecks.test_exit_code_error(self)
642
642
643 @skip_win32
643 @skip_win32
644 def test_exit_code_signal(self):
644 def test_exit_code_signal(self):
645 ExitCodeChecks.test_exit_code_signal(self)
645 ExitCodeChecks.test_exit_code_signal(self)
646
646
647 class TestModules(tt.TempFileMixin):
647 class TestModules(tt.TempFileMixin):
648 def test_extraneous_loads(self):
648 def test_extraneous_loads(self):
649 """Test we're not loading modules on startup that we shouldn't.
649 """Test we're not loading modules on startup that we shouldn't.
650 """
650 """
651 self.mktmp("import sys\n"
651 self.mktmp("import sys\n"
652 "print('numpy' in sys.modules)\n"
652 "print('numpy' in sys.modules)\n"
653 "print('ipyparallel' in sys.modules)\n"
653 "print('ipyparallel' in sys.modules)\n"
654 "print('ipykernel' in sys.modules)\n"
654 "print('ipykernel' in sys.modules)\n"
655 )
655 )
656 out = "False\nFalse\nFalse\n"
656 out = "False\nFalse\nFalse\n"
657 tt.ipexec_validate(self.fname, out)
657 tt.ipexec_validate(self.fname, out)
658
658
659 class Negator(ast.NodeTransformer):
659 class Negator(ast.NodeTransformer):
660 """Negates all number literals in an AST."""
660 """Negates all number literals in an AST."""
661
661
662 # for python 3.7 and earlier
662 # for python 3.7 and earlier
663 def visit_Num(self, node):
663 def visit_Num(self, node):
664 node.n = -node.n
664 node.n = -node.n
665 return node
665 return node
666
666
667 # for python 3.8+
667 # for python 3.8+
668 def visit_Constant(self, node):
668 def visit_Constant(self, node):
669 if isinstance(node.value, int):
669 if isinstance(node.value, int):
670 return self.visit_Num(node)
670 return self.visit_Num(node)
671 return node
671 return node
672
672
673 class TestAstTransform(unittest.TestCase):
673 class TestAstTransform(unittest.TestCase):
674 def setUp(self):
674 def setUp(self):
675 self.negator = Negator()
675 self.negator = Negator()
676 ip.ast_transformers.append(self.negator)
676 ip.ast_transformers.append(self.negator)
677
677
678 def tearDown(self):
678 def tearDown(self):
679 ip.ast_transformers.remove(self.negator)
679 ip.ast_transformers.remove(self.negator)
680
680
681 def test_run_cell(self):
681 def test_run_cell(self):
682 with tt.AssertPrints('-34'):
682 with tt.AssertPrints('-34'):
683 ip.run_cell('print (12 + 22)')
683 ip.run_cell('print (12 + 22)')
684
684
685 # A named reference to a number shouldn't be transformed.
685 # A named reference to a number shouldn't be transformed.
686 ip.user_ns['n'] = 55
686 ip.user_ns['n'] = 55
687 with tt.AssertNotPrints('-55'):
687 with tt.AssertNotPrints('-55'):
688 ip.run_cell('print (n)')
688 ip.run_cell('print (n)')
689
689
690 def test_timeit(self):
690 def test_timeit(self):
691 called = set()
691 called = set()
692 def f(x):
692 def f(x):
693 called.add(x)
693 called.add(x)
694 ip.push({'f':f})
694 ip.push({'f':f})
695
695
696 with tt.AssertPrints("std. dev. of"):
696 with tt.AssertPrints("std. dev. of"):
697 ip.run_line_magic("timeit", "-n1 f(1)")
697 ip.run_line_magic("timeit", "-n1 f(1)")
698 self.assertEqual(called, {-1})
698 self.assertEqual(called, {-1})
699 called.clear()
699 called.clear()
700
700
701 with tt.AssertPrints("std. dev. of"):
701 with tt.AssertPrints("std. dev. of"):
702 ip.run_cell_magic("timeit", "-n1 f(2)", "f(3)")
702 ip.run_cell_magic("timeit", "-n1 f(2)", "f(3)")
703 self.assertEqual(called, {-2, -3})
703 self.assertEqual(called, {-2, -3})
704
704
705 def test_time(self):
705 def test_time(self):
706 called = []
706 called = []
707 def f(x):
707 def f(x):
708 called.append(x)
708 called.append(x)
709 ip.push({'f':f})
709 ip.push({'f':f})
710
710
711 # Test with an expression
711 # Test with an expression
712 with tt.AssertPrints("Wall time: "):
712 with tt.AssertPrints("Wall time: "):
713 ip.run_line_magic("time", "f(5+9)")
713 ip.run_line_magic("time", "f(5+9)")
714 self.assertEqual(called, [-14])
714 self.assertEqual(called, [-14])
715 called[:] = []
715 called[:] = []
716
716
717 # Test with a statement (different code path)
717 # Test with a statement (different code path)
718 with tt.AssertPrints("Wall time: "):
718 with tt.AssertPrints("Wall time: "):
719 ip.run_line_magic("time", "a = f(-3 + -2)")
719 ip.run_line_magic("time", "a = f(-3 + -2)")
720 self.assertEqual(called, [5])
720 self.assertEqual(called, [5])
721
721
722 def test_macro(self):
722 def test_macro(self):
723 ip.push({'a':10})
723 ip.push({'a':10})
724 # The AST transformation makes this do a+=-1
724 # The AST transformation makes this do a+=-1
725 ip.define_macro("amacro", "a+=1\nprint(a)")
725 ip.define_macro("amacro", "a+=1\nprint(a)")
726
726
727 with tt.AssertPrints("9"):
727 with tt.AssertPrints("9"):
728 ip.run_cell("amacro")
728 ip.run_cell("amacro")
729 with tt.AssertPrints("8"):
729 with tt.AssertPrints("8"):
730 ip.run_cell("amacro")
730 ip.run_cell("amacro")
731
731
732 class TestMiscTransform(unittest.TestCase):
732 class TestMiscTransform(unittest.TestCase):
733
733
734
734
735 def test_transform_only_once(self):
735 def test_transform_only_once(self):
736 cleanup = 0
736 cleanup = 0
737 line_t = 0
737 line_t = 0
738 def count_cleanup(lines):
738 def count_cleanup(lines):
739 nonlocal cleanup
739 nonlocal cleanup
740 cleanup += 1
740 cleanup += 1
741 return lines
741 return lines
742
742
743 def count_line_t(lines):
743 def count_line_t(lines):
744 nonlocal line_t
744 nonlocal line_t
745 line_t += 1
745 line_t += 1
746 return lines
746 return lines
747
747
748 ip.input_transformer_manager.cleanup_transforms.append(count_cleanup)
748 ip.input_transformer_manager.cleanup_transforms.append(count_cleanup)
749 ip.input_transformer_manager.line_transforms.append(count_line_t)
749 ip.input_transformer_manager.line_transforms.append(count_line_t)
750
750
751 ip.run_cell('1')
751 ip.run_cell('1')
752
752
753 assert cleanup == 1
753 assert cleanup == 1
754 assert line_t == 1
754 assert line_t == 1
755
755
756 class IntegerWrapper(ast.NodeTransformer):
756 class IntegerWrapper(ast.NodeTransformer):
757 """Wraps all integers in a call to Integer()"""
757 """Wraps all integers in a call to Integer()"""
758
758
759 # for Python 3.7 and earlier
759 # for Python 3.7 and earlier
760
760
761 # for Python 3.7 and earlier
761 # for Python 3.7 and earlier
762 def visit_Num(self, node):
762 def visit_Num(self, node):
763 if isinstance(node.n, int):
763 if isinstance(node.n, int):
764 return ast.Call(func=ast.Name(id='Integer', ctx=ast.Load()),
764 return ast.Call(func=ast.Name(id='Integer', ctx=ast.Load()),
765 args=[node], keywords=[])
765 args=[node], keywords=[])
766 return node
766 return node
767
767
768 # For Python 3.8+
768 # For Python 3.8+
769 def visit_Constant(self, node):
769 def visit_Constant(self, node):
770 if isinstance(node.value, int):
770 if isinstance(node.value, int):
771 return self.visit_Num(node)
771 return self.visit_Num(node)
772 return node
772 return node
773
773
774
774
775 class TestAstTransform2(unittest.TestCase):
775 class TestAstTransform2(unittest.TestCase):
776 def setUp(self):
776 def setUp(self):
777 self.intwrapper = IntegerWrapper()
777 self.intwrapper = IntegerWrapper()
778 ip.ast_transformers.append(self.intwrapper)
778 ip.ast_transformers.append(self.intwrapper)
779
779
780 self.calls = []
780 self.calls = []
781 def Integer(*args):
781 def Integer(*args):
782 self.calls.append(args)
782 self.calls.append(args)
783 return args
783 return args
784 ip.push({"Integer": Integer})
784 ip.push({"Integer": Integer})
785
785
786 def tearDown(self):
786 def tearDown(self):
787 ip.ast_transformers.remove(self.intwrapper)
787 ip.ast_transformers.remove(self.intwrapper)
788 del ip.user_ns['Integer']
788 del ip.user_ns['Integer']
789
789
790 def test_run_cell(self):
790 def test_run_cell(self):
791 ip.run_cell("n = 2")
791 ip.run_cell("n = 2")
792 self.assertEqual(self.calls, [(2,)])
792 self.assertEqual(self.calls, [(2,)])
793
793
794 # This shouldn't throw an error
794 # This shouldn't throw an error
795 ip.run_cell("o = 2.0")
795 ip.run_cell("o = 2.0")
796 self.assertEqual(ip.user_ns['o'], 2.0)
796 self.assertEqual(ip.user_ns['o'], 2.0)
797
797
798 def test_timeit(self):
798 def test_timeit(self):
799 called = set()
799 called = set()
800 def f(x):
800 def f(x):
801 called.add(x)
801 called.add(x)
802 ip.push({'f':f})
802 ip.push({'f':f})
803
803
804 with tt.AssertPrints("std. dev. of"):
804 with tt.AssertPrints("std. dev. of"):
805 ip.run_line_magic("timeit", "-n1 f(1)")
805 ip.run_line_magic("timeit", "-n1 f(1)")
806 self.assertEqual(called, {(1,)})
806 self.assertEqual(called, {(1,)})
807 called.clear()
807 called.clear()
808
808
809 with tt.AssertPrints("std. dev. of"):
809 with tt.AssertPrints("std. dev. of"):
810 ip.run_cell_magic("timeit", "-n1 f(2)", "f(3)")
810 ip.run_cell_magic("timeit", "-n1 f(2)", "f(3)")
811 self.assertEqual(called, {(2,), (3,)})
811 self.assertEqual(called, {(2,), (3,)})
812
812
813 class ErrorTransformer(ast.NodeTransformer):
813 class ErrorTransformer(ast.NodeTransformer):
814 """Throws an error when it sees a number."""
814 """Throws an error when it sees a number."""
815
815
816 # for Python 3.7 and earlier
816 # for Python 3.7 and earlier
817 def visit_Num(self, node):
817 def visit_Num(self, node):
818 raise ValueError("test")
818 raise ValueError("test")
819
819
820 # for Python 3.8+
820 # for Python 3.8+
821 def visit_Constant(self, node):
821 def visit_Constant(self, node):
822 if isinstance(node.value, int):
822 if isinstance(node.value, int):
823 return self.visit_Num(node)
823 return self.visit_Num(node)
824 return node
824 return node
825
825
826
826
827 class TestAstTransformError(unittest.TestCase):
827 class TestAstTransformError(unittest.TestCase):
828 def test_unregistering(self):
828 def test_unregistering(self):
829 err_transformer = ErrorTransformer()
829 err_transformer = ErrorTransformer()
830 ip.ast_transformers.append(err_transformer)
830 ip.ast_transformers.append(err_transformer)
831
831
832 with self.assertWarnsRegex(UserWarning, "It will be unregistered"):
832 with self.assertWarnsRegex(UserWarning, "It will be unregistered"):
833 ip.run_cell("1 + 2")
833 ip.run_cell("1 + 2")
834
834
835 # This should have been removed.
835 # This should have been removed.
836 self.assertNotIn(err_transformer, ip.ast_transformers)
836 self.assertNotIn(err_transformer, ip.ast_transformers)
837
837
838
838
839 class StringRejector(ast.NodeTransformer):
839 class StringRejector(ast.NodeTransformer):
840 """Throws an InputRejected when it sees a string literal.
840 """Throws an InputRejected when it sees a string literal.
841
841
842 Used to verify that NodeTransformers can signal that a piece of code should
842 Used to verify that NodeTransformers can signal that a piece of code should
843 not be executed by throwing an InputRejected.
843 not be executed by throwing an InputRejected.
844 """
844 """
845
845
846 #for python 3.7 and earlier
846 #for python 3.7 and earlier
847 def visit_Str(self, node):
847 def visit_Str(self, node):
848 raise InputRejected("test")
848 raise InputRejected("test")
849
849
850 # 3.8 only
850 # 3.8 only
851 def visit_Constant(self, node):
851 def visit_Constant(self, node):
852 if isinstance(node.value, str):
852 if isinstance(node.value, str):
853 raise InputRejected("test")
853 raise InputRejected("test")
854 return node
854 return node
855
855
856
856
857 class TestAstTransformInputRejection(unittest.TestCase):
857 class TestAstTransformInputRejection(unittest.TestCase):
858
858
859 def setUp(self):
859 def setUp(self):
860 self.transformer = StringRejector()
860 self.transformer = StringRejector()
861 ip.ast_transformers.append(self.transformer)
861 ip.ast_transformers.append(self.transformer)
862
862
863 def tearDown(self):
863 def tearDown(self):
864 ip.ast_transformers.remove(self.transformer)
864 ip.ast_transformers.remove(self.transformer)
865
865
866 def test_input_rejection(self):
866 def test_input_rejection(self):
867 """Check that NodeTransformers can reject input."""
867 """Check that NodeTransformers can reject input."""
868
868
869 expect_exception_tb = tt.AssertPrints("InputRejected: test")
869 expect_exception_tb = tt.AssertPrints("InputRejected: test")
870 expect_no_cell_output = tt.AssertNotPrints("'unsafe'", suppress=False)
870 expect_no_cell_output = tt.AssertNotPrints("'unsafe'", suppress=False)
871
871
872 # Run the same check twice to verify that the transformer is not
872 # Run the same check twice to verify that the transformer is not
873 # disabled after raising.
873 # disabled after raising.
874 with expect_exception_tb, expect_no_cell_output:
874 with expect_exception_tb, expect_no_cell_output:
875 ip.run_cell("'unsafe'")
875 ip.run_cell("'unsafe'")
876
876
877 with expect_exception_tb, expect_no_cell_output:
877 with expect_exception_tb, expect_no_cell_output:
878 res = ip.run_cell("'unsafe'")
878 res = ip.run_cell("'unsafe'")
879
879
880 self.assertIsInstance(res.error_before_exec, InputRejected)
880 self.assertIsInstance(res.error_before_exec, InputRejected)
881
881
882 def test__IPYTHON__():
882 def test__IPYTHON__():
883 # This shouldn't raise a NameError, that's all
883 # This shouldn't raise a NameError, that's all
884 __IPYTHON__
884 __IPYTHON__
885
885
886
886
887 class DummyRepr(object):
887 class DummyRepr(object):
888 def __repr__(self):
888 def __repr__(self):
889 return "DummyRepr"
889 return "DummyRepr"
890
890
891 def _repr_html_(self):
891 def _repr_html_(self):
892 return "<b>dummy</b>"
892 return "<b>dummy</b>"
893
893
894 def _repr_javascript_(self):
894 def _repr_javascript_(self):
895 return "console.log('hi');", {'key': 'value'}
895 return "console.log('hi');", {'key': 'value'}
896
896
897
897
898 def test_user_variables():
898 def test_user_variables():
899 # enable all formatters
899 # enable all formatters
900 ip.display_formatter.active_types = ip.display_formatter.format_types
900 ip.display_formatter.active_types = ip.display_formatter.format_types
901
901
902 ip.user_ns['dummy'] = d = DummyRepr()
902 ip.user_ns['dummy'] = d = DummyRepr()
903 keys = {'dummy', 'doesnotexist'}
903 keys = {'dummy', 'doesnotexist'}
904 r = ip.user_expressions({ key:key for key in keys})
904 r = ip.user_expressions({ key:key for key in keys})
905
905
906 assert keys == set(r.keys())
906 assert keys == set(r.keys())
907 dummy = r["dummy"]
907 dummy = r["dummy"]
908 assert {"status", "data", "metadata"} == set(dummy.keys())
908 assert {"status", "data", "metadata"} == set(dummy.keys())
909 assert dummy["status"] == "ok"
909 assert dummy["status"] == "ok"
910 data = dummy["data"]
910 data = dummy["data"]
911 metadata = dummy["metadata"]
911 metadata = dummy["metadata"]
912 assert data.get("text/html") == d._repr_html_()
912 assert data.get("text/html") == d._repr_html_()
913 js, jsmd = d._repr_javascript_()
913 js, jsmd = d._repr_javascript_()
914 assert data.get("application/javascript") == js
914 assert data.get("application/javascript") == js
915 assert metadata.get("application/javascript") == jsmd
915 assert metadata.get("application/javascript") == jsmd
916
916
917 dne = r["doesnotexist"]
917 dne = r["doesnotexist"]
918 assert dne["status"] == "error"
918 assert dne["status"] == "error"
919 assert dne["ename"] == "NameError"
919 assert dne["ename"] == "NameError"
920
920
921 # back to text only
921 # back to text only
922 ip.display_formatter.active_types = ['text/plain']
922 ip.display_formatter.active_types = ['text/plain']
923
923
924 def test_user_expression():
924 def test_user_expression():
925 # enable all formatters
925 # enable all formatters
926 ip.display_formatter.active_types = ip.display_formatter.format_types
926 ip.display_formatter.active_types = ip.display_formatter.format_types
927 query = {
927 query = {
928 'a' : '1 + 2',
928 'a' : '1 + 2',
929 'b' : '1/0',
929 'b' : '1/0',
930 }
930 }
931 r = ip.user_expressions(query)
931 r = ip.user_expressions(query)
932 import pprint
932 import pprint
933 pprint.pprint(r)
933 pprint.pprint(r)
934 assert set(r.keys()) == set(query.keys())
934 assert set(r.keys()) == set(query.keys())
935 a = r["a"]
935 a = r["a"]
936 assert {"status", "data", "metadata"} == set(a.keys())
936 assert {"status", "data", "metadata"} == set(a.keys())
937 assert a["status"] == "ok"
937 assert a["status"] == "ok"
938 data = a["data"]
938 data = a["data"]
939 metadata = a["metadata"]
939 metadata = a["metadata"]
940 assert data.get("text/plain") == "3"
940 assert data.get("text/plain") == "3"
941
941
942 b = r["b"]
942 b = r["b"]
943 assert b["status"] == "error"
943 assert b["status"] == "error"
944 assert b["ename"] == "ZeroDivisionError"
944 assert b["ename"] == "ZeroDivisionError"
945
945
946 # back to text only
946 # back to text only
947 ip.display_formatter.active_types = ['text/plain']
947 ip.display_formatter.active_types = ['text/plain']
948
948
949
949
950 class TestSyntaxErrorTransformer(unittest.TestCase):
950 class TestSyntaxErrorTransformer(unittest.TestCase):
951 """Check that SyntaxError raised by an input transformer is handled by run_cell()"""
951 """Check that SyntaxError raised by an input transformer is handled by run_cell()"""
952
952
953 @staticmethod
953 @staticmethod
954 def transformer(lines):
954 def transformer(lines):
955 for line in lines:
955 for line in lines:
956 pos = line.find('syntaxerror')
956 pos = line.find('syntaxerror')
957 if pos >= 0:
957 if pos >= 0:
958 e = SyntaxError('input contains "syntaxerror"')
958 e = SyntaxError('input contains "syntaxerror"')
959 e.text = line
959 e.text = line
960 e.offset = pos + 1
960 e.offset = pos + 1
961 raise e
961 raise e
962 return lines
962 return lines
963
963
964 def setUp(self):
964 def setUp(self):
965 ip.input_transformers_post.append(self.transformer)
965 ip.input_transformers_post.append(self.transformer)
966
966
967 def tearDown(self):
967 def tearDown(self):
968 ip.input_transformers_post.remove(self.transformer)
968 ip.input_transformers_post.remove(self.transformer)
969
969
970 def test_syntaxerror_input_transformer(self):
970 def test_syntaxerror_input_transformer(self):
971 with tt.AssertPrints('1234'):
971 with tt.AssertPrints('1234'):
972 ip.run_cell('1234')
972 ip.run_cell('1234')
973 with tt.AssertPrints('SyntaxError: invalid syntax'):
973 with tt.AssertPrints('SyntaxError: invalid syntax'):
974 ip.run_cell('1 2 3') # plain python syntax error
974 ip.run_cell('1 2 3') # plain python syntax error
975 with tt.AssertPrints('SyntaxError: input contains "syntaxerror"'):
975 with tt.AssertPrints('SyntaxError: input contains "syntaxerror"'):
976 ip.run_cell('2345 # syntaxerror') # input transformer syntax error
976 ip.run_cell('2345 # syntaxerror') # input transformer syntax error
977 with tt.AssertPrints('3456'):
977 with tt.AssertPrints('3456'):
978 ip.run_cell('3456')
978 ip.run_cell('3456')
979
979
980
980
981 class TestWarningSuppression(unittest.TestCase):
981 class TestWarningSuppression(unittest.TestCase):
982 def test_warning_suppression(self):
982 def test_warning_suppression(self):
983 ip.run_cell("import warnings")
983 ip.run_cell("import warnings")
984 try:
984 try:
985 with self.assertWarnsRegex(UserWarning, "asdf"):
985 with self.assertWarnsRegex(UserWarning, "asdf"):
986 ip.run_cell("warnings.warn('asdf')")
986 ip.run_cell("warnings.warn('asdf')")
987 # Here's the real test -- if we run that again, we should get the
987 # Here's the real test -- if we run that again, we should get the
988 # warning again. Traditionally, each warning was only issued once per
988 # warning again. Traditionally, each warning was only issued once per
989 # IPython session (approximately), even if the user typed in new and
989 # IPython session (approximately), even if the user typed in new and
990 # different code that should have also triggered the warning, leading
990 # different code that should have also triggered the warning, leading
991 # to much confusion.
991 # to much confusion.
992 with self.assertWarnsRegex(UserWarning, "asdf"):
992 with self.assertWarnsRegex(UserWarning, "asdf"):
993 ip.run_cell("warnings.warn('asdf')")
993 ip.run_cell("warnings.warn('asdf')")
994 finally:
994 finally:
995 ip.run_cell("del warnings")
995 ip.run_cell("del warnings")
996
996
997
997
998 def test_deprecation_warning(self):
998 def test_deprecation_warning(self):
999 ip.run_cell("""
999 ip.run_cell("""
1000 import warnings
1000 import warnings
1001 def wrn():
1001 def wrn():
1002 warnings.warn(
1002 warnings.warn(
1003 "I AM A WARNING",
1003 "I AM A WARNING",
1004 DeprecationWarning
1004 DeprecationWarning
1005 )
1005 )
1006 """)
1006 """)
1007 try:
1007 try:
1008 with self.assertWarnsRegex(DeprecationWarning, "I AM A WARNING"):
1008 with self.assertWarnsRegex(DeprecationWarning, "I AM A WARNING"):
1009 ip.run_cell("wrn()")
1009 ip.run_cell("wrn()")
1010 finally:
1010 finally:
1011 ip.run_cell("del warnings")
1011 ip.run_cell("del warnings")
1012 ip.run_cell("del wrn")
1012 ip.run_cell("del wrn")
1013
1013
1014
1014
1015 class TestImportNoDeprecate(tt.TempFileMixin):
1015 class TestImportNoDeprecate(tt.TempFileMixin):
1016
1016
1017 def setUp(self):
1017 def setUp(self):
1018 """Make a valid python temp file."""
1018 """Make a valid python temp file."""
1019 self.mktmp("""
1019 self.mktmp("""
1020 import warnings
1020 import warnings
1021 def wrn():
1021 def wrn():
1022 warnings.warn(
1022 warnings.warn(
1023 "I AM A WARNING",
1023 "I AM A WARNING",
1024 DeprecationWarning
1024 DeprecationWarning
1025 )
1025 )
1026 """)
1026 """)
1027 super().setUp()
1027 super().setUp()
1028
1028
1029 def test_no_dep(self):
1029 def test_no_dep(self):
1030 """
1030 """
1031 No deprecation warning should be raised from imported functions
1031 No deprecation warning should be raised from imported functions
1032 """
1032 """
1033 ip.run_cell("from {} import wrn".format(self.fname))
1033 ip.run_cell("from {} import wrn".format(self.fname))
1034
1034
1035 with tt.AssertNotPrints("I AM A WARNING"):
1035 with tt.AssertNotPrints("I AM A WARNING"):
1036 ip.run_cell("wrn()")
1036 ip.run_cell("wrn()")
1037 ip.run_cell("del wrn")
1037 ip.run_cell("del wrn")
1038
1038
1039
1039
1040 def test_custom_exc_count():
1040 def test_custom_exc_count():
1041 hook = mock.Mock(return_value=None)
1041 hook = mock.Mock(return_value=None)
1042 ip.set_custom_exc((SyntaxError,), hook)
1042 ip.set_custom_exc((SyntaxError,), hook)
1043 before = ip.execution_count
1043 before = ip.execution_count
1044 ip.run_cell("def foo()", store_history=True)
1044 ip.run_cell("def foo()", store_history=True)
1045 # restore default excepthook
1045 # restore default excepthook
1046 ip.set_custom_exc((), None)
1046 ip.set_custom_exc((), None)
1047 assert hook.call_count == 1
1047 assert hook.call_count == 1
1048 assert ip.execution_count == before + 1
1048 assert ip.execution_count == before + 1
1049
1049
1050
1050
1051 def test_run_cell_async():
1051 def test_run_cell_async():
1052 loop = asyncio.get_event_loop_policy().get_event_loop()
1053 ip.run_cell("import asyncio")
1052 ip.run_cell("import asyncio")
1054 coro = ip.run_cell_async("await asyncio.sleep(0.01)\n5")
1053 coro = ip.run_cell_async("await asyncio.sleep(0.01)\n5")
1055 assert asyncio.iscoroutine(coro)
1054 assert asyncio.iscoroutine(coro)
1055 loop = asyncio.new_event_loop()
1056 result = loop.run_until_complete(coro)
1056 result = loop.run_until_complete(coro)
1057 assert isinstance(result, interactiveshell.ExecutionResult)
1057 assert isinstance(result, interactiveshell.ExecutionResult)
1058 assert result.result == 5
1058 assert result.result == 5
1059
1059
1060
1060
1061 def test_run_cell_await():
1061 def test_run_cell_await():
1062 ip.run_cell("import asyncio")
1062 ip.run_cell("import asyncio")
1063 result = ip.run_cell("await asyncio.sleep(0.01); 10")
1063 result = ip.run_cell("await asyncio.sleep(0.01); 10")
1064 assert ip.user_ns["_"] == 10
1064 assert ip.user_ns["_"] == 10
1065
1065
1066
1066
1067 def test_run_cell_asyncio_run():
1067 def test_run_cell_asyncio_run():
1068 ip.run_cell("import asyncio")
1068 ip.run_cell("import asyncio")
1069 result = ip.run_cell("await asyncio.sleep(0.01); 1")
1069 result = ip.run_cell("await asyncio.sleep(0.01); 1")
1070 assert ip.user_ns["_"] == 1
1070 assert ip.user_ns["_"] == 1
1071 result = ip.run_cell("asyncio.run(asyncio.sleep(0.01)); 2")
1071 result = ip.run_cell("asyncio.run(asyncio.sleep(0.01)); 2")
1072 assert ip.user_ns["_"] == 2
1072 assert ip.user_ns["_"] == 2
1073 result = ip.run_cell("await asyncio.sleep(0.01); 3")
1073 result = ip.run_cell("await asyncio.sleep(0.01); 3")
1074 assert ip.user_ns["_"] == 3
1074 assert ip.user_ns["_"] == 3
1075
1075
1076
1076
1077 def test_should_run_async():
1077 def test_should_run_async():
1078 assert not ip.should_run_async("a = 5")
1078 assert not ip.should_run_async("a = 5")
1079 assert ip.should_run_async("await x")
1079 assert ip.should_run_async("await x")
1080 assert ip.should_run_async("import asyncio; await asyncio.sleep(1)")
1080 assert ip.should_run_async("import asyncio; await asyncio.sleep(1)")
1081
1081
1082
1082
1083 def test_set_custom_completer():
1083 def test_set_custom_completer():
1084 num_completers = len(ip.Completer.matchers)
1084 num_completers = len(ip.Completer.matchers)
1085
1085
1086 def foo(*args, **kwargs):
1086 def foo(*args, **kwargs):
1087 return "I'm a completer!"
1087 return "I'm a completer!"
1088
1088
1089 ip.set_custom_completer(foo, 0)
1089 ip.set_custom_completer(foo, 0)
1090
1090
1091 # check that we've really added a new completer
1091 # check that we've really added a new completer
1092 assert len(ip.Completer.matchers) == num_completers + 1
1092 assert len(ip.Completer.matchers) == num_completers + 1
1093
1093
1094 # check that the first completer is the function we defined
1094 # check that the first completer is the function we defined
1095 assert ip.Completer.matchers[0]() == "I'm a completer!"
1095 assert ip.Completer.matchers[0]() == "I'm a completer!"
1096
1096
1097 # clean up
1097 # clean up
1098 ip.Completer.custom_matchers.pop()
1098 ip.Completer.custom_matchers.pop()
@@ -1,1358 +1,1366 b''
1 # -*- coding: utf-8 -*-
1 # -*- coding: utf-8 -*-
2 """Tests for various magic functions."""
2 """Tests for various magic functions."""
3
3
4 import asyncio
4 import asyncio
5 import gc
5 import gc
6 import io
6 import io
7 import os
7 import os
8 import re
8 import re
9 import shlex
9 import shlex
10 import sys
10 import sys
11 import warnings
11 import warnings
12 from importlib import invalidate_caches
12 from importlib import invalidate_caches
13 from io import StringIO
13 from io import StringIO
14 from pathlib import Path
14 from pathlib import Path
15 from textwrap import dedent
15 from textwrap import dedent
16 from unittest import TestCase, mock
16 from unittest import TestCase, mock
17
17
18 import pytest
18 import pytest
19
19
20 from IPython import get_ipython
20 from IPython import get_ipython
21 from IPython.core import magic
21 from IPython.core import magic
22 from IPython.core.error import UsageError
22 from IPython.core.error import UsageError
23 from IPython.core.magic import (
23 from IPython.core.magic import (
24 Magics,
24 Magics,
25 cell_magic,
25 cell_magic,
26 line_magic,
26 line_magic,
27 magics_class,
27 magics_class,
28 register_cell_magic,
28 register_cell_magic,
29 register_line_magic,
29 register_line_magic,
30 )
30 )
31 from IPython.core.magics import code, execution, logging, osm, script
31 from IPython.core.magics import code, execution, logging, osm, script
32 from IPython.testing import decorators as dec
32 from IPython.testing import decorators as dec
33 from IPython.testing import tools as tt
33 from IPython.testing import tools as tt
34 from IPython.utils.io import capture_output
34 from IPython.utils.io import capture_output
35 from IPython.utils.process import find_cmd
35 from IPython.utils.process import find_cmd
36 from IPython.utils.tempdir import TemporaryDirectory, TemporaryWorkingDirectory
36 from IPython.utils.tempdir import TemporaryDirectory, TemporaryWorkingDirectory
37
37
38 from .test_debugger import PdbTestInput
38 from .test_debugger import PdbTestInput
39
39
40
40
41 @magic.magics_class
41 @magic.magics_class
42 class DummyMagics(magic.Magics): pass
42 class DummyMagics(magic.Magics): pass
43
43
44 def test_extract_code_ranges():
44 def test_extract_code_ranges():
45 instr = "1 3 5-6 7-9 10:15 17: :10 10- -13 :"
45 instr = "1 3 5-6 7-9 10:15 17: :10 10- -13 :"
46 expected = [
46 expected = [
47 (0, 1),
47 (0, 1),
48 (2, 3),
48 (2, 3),
49 (4, 6),
49 (4, 6),
50 (6, 9),
50 (6, 9),
51 (9, 14),
51 (9, 14),
52 (16, None),
52 (16, None),
53 (None, 9),
53 (None, 9),
54 (9, None),
54 (9, None),
55 (None, 13),
55 (None, 13),
56 (None, None),
56 (None, None),
57 ]
57 ]
58 actual = list(code.extract_code_ranges(instr))
58 actual = list(code.extract_code_ranges(instr))
59 assert actual == expected
59 assert actual == expected
60
60
61 def test_extract_symbols():
61 def test_extract_symbols():
62 source = """import foo\na = 10\ndef b():\n return 42\n\n\nclass A: pass\n\n\n"""
62 source = """import foo\na = 10\ndef b():\n return 42\n\n\nclass A: pass\n\n\n"""
63 symbols_args = ["a", "b", "A", "A,b", "A,a", "z"]
63 symbols_args = ["a", "b", "A", "A,b", "A,a", "z"]
64 expected = [([], ['a']),
64 expected = [([], ['a']),
65 (["def b():\n return 42\n"], []),
65 (["def b():\n return 42\n"], []),
66 (["class A: pass\n"], []),
66 (["class A: pass\n"], []),
67 (["class A: pass\n", "def b():\n return 42\n"], []),
67 (["class A: pass\n", "def b():\n return 42\n"], []),
68 (["class A: pass\n"], ['a']),
68 (["class A: pass\n"], ['a']),
69 ([], ['z'])]
69 ([], ['z'])]
70 for symbols, exp in zip(symbols_args, expected):
70 for symbols, exp in zip(symbols_args, expected):
71 assert code.extract_symbols(source, symbols) == exp
71 assert code.extract_symbols(source, symbols) == exp
72
72
73
73
74 def test_extract_symbols_raises_exception_with_non_python_code():
74 def test_extract_symbols_raises_exception_with_non_python_code():
75 source = ("=begin A Ruby program :)=end\n"
75 source = ("=begin A Ruby program :)=end\n"
76 "def hello\n"
76 "def hello\n"
77 "puts 'Hello world'\n"
77 "puts 'Hello world'\n"
78 "end")
78 "end")
79 with pytest.raises(SyntaxError):
79 with pytest.raises(SyntaxError):
80 code.extract_symbols(source, "hello")
80 code.extract_symbols(source, "hello")
81
81
82
82
83 def test_magic_not_found():
83 def test_magic_not_found():
84 # magic not found raises UsageError
84 # magic not found raises UsageError
85 with pytest.raises(UsageError):
85 with pytest.raises(UsageError):
86 _ip.magic('doesntexist')
86 _ip.magic('doesntexist')
87
87
88 # ensure result isn't success when a magic isn't found
88 # ensure result isn't success when a magic isn't found
89 result = _ip.run_cell('%doesntexist')
89 result = _ip.run_cell('%doesntexist')
90 assert isinstance(result.error_in_exec, UsageError)
90 assert isinstance(result.error_in_exec, UsageError)
91
91
92
92
93 def test_cell_magic_not_found():
93 def test_cell_magic_not_found():
94 # magic not found raises UsageError
94 # magic not found raises UsageError
95 with pytest.raises(UsageError):
95 with pytest.raises(UsageError):
96 _ip.run_cell_magic('doesntexist', 'line', 'cell')
96 _ip.run_cell_magic('doesntexist', 'line', 'cell')
97
97
98 # ensure result isn't success when a magic isn't found
98 # ensure result isn't success when a magic isn't found
99 result = _ip.run_cell('%%doesntexist')
99 result = _ip.run_cell('%%doesntexist')
100 assert isinstance(result.error_in_exec, UsageError)
100 assert isinstance(result.error_in_exec, UsageError)
101
101
102
102
103 def test_magic_error_status():
103 def test_magic_error_status():
104 def fail(shell):
104 def fail(shell):
105 1/0
105 1/0
106 _ip.register_magic_function(fail)
106 _ip.register_magic_function(fail)
107 result = _ip.run_cell('%fail')
107 result = _ip.run_cell('%fail')
108 assert isinstance(result.error_in_exec, ZeroDivisionError)
108 assert isinstance(result.error_in_exec, ZeroDivisionError)
109
109
110
110
111 def test_config():
111 def test_config():
112 """ test that config magic does not raise
112 """ test that config magic does not raise
113 can happen if Configurable init is moved too early into
113 can happen if Configurable init is moved too early into
114 Magics.__init__ as then a Config object will be registered as a
114 Magics.__init__ as then a Config object will be registered as a
115 magic.
115 magic.
116 """
116 """
117 ## should not raise.
117 ## should not raise.
118 _ip.magic('config')
118 _ip.magic('config')
119
119
120 def test_config_available_configs():
120 def test_config_available_configs():
121 """ test that config magic prints available configs in unique and
121 """ test that config magic prints available configs in unique and
122 sorted order. """
122 sorted order. """
123 with capture_output() as captured:
123 with capture_output() as captured:
124 _ip.magic('config')
124 _ip.magic('config')
125
125
126 stdout = captured.stdout
126 stdout = captured.stdout
127 config_classes = stdout.strip().split('\n')[1:]
127 config_classes = stdout.strip().split('\n')[1:]
128 assert config_classes == sorted(set(config_classes))
128 assert config_classes == sorted(set(config_classes))
129
129
130 def test_config_print_class():
130 def test_config_print_class():
131 """ test that config with a classname prints the class's options. """
131 """ test that config with a classname prints the class's options. """
132 with capture_output() as captured:
132 with capture_output() as captured:
133 _ip.magic('config TerminalInteractiveShell')
133 _ip.magic('config TerminalInteractiveShell')
134
134
135 stdout = captured.stdout
135 stdout = captured.stdout
136 assert re.match(
136 assert re.match(
137 "TerminalInteractiveShell.* options", stdout.splitlines()[0]
137 "TerminalInteractiveShell.* options", stdout.splitlines()[0]
138 ), f"{stdout}\n\n1st line of stdout not like 'TerminalInteractiveShell.* options'"
138 ), f"{stdout}\n\n1st line of stdout not like 'TerminalInteractiveShell.* options'"
139
139
140
140
141 def test_rehashx():
141 def test_rehashx():
142 # clear up everything
142 # clear up everything
143 _ip.alias_manager.clear_aliases()
143 _ip.alias_manager.clear_aliases()
144 del _ip.db['syscmdlist']
144 del _ip.db['syscmdlist']
145
145
146 _ip.magic('rehashx')
146 _ip.magic('rehashx')
147 # Practically ALL ipython development systems will have more than 10 aliases
147 # Practically ALL ipython development systems will have more than 10 aliases
148
148
149 assert len(_ip.alias_manager.aliases) > 10
149 assert len(_ip.alias_manager.aliases) > 10
150 for name, cmd in _ip.alias_manager.aliases:
150 for name, cmd in _ip.alias_manager.aliases:
151 # we must strip dots from alias names
151 # we must strip dots from alias names
152 assert "." not in name
152 assert "." not in name
153
153
154 # rehashx must fill up syscmdlist
154 # rehashx must fill up syscmdlist
155 scoms = _ip.db['syscmdlist']
155 scoms = _ip.db['syscmdlist']
156 assert len(scoms) > 10
156 assert len(scoms) > 10
157
157
158
158
159 def test_magic_parse_options():
159 def test_magic_parse_options():
160 """Test that we don't mangle paths when parsing magic options."""
160 """Test that we don't mangle paths when parsing magic options."""
161 ip = get_ipython()
161 ip = get_ipython()
162 path = 'c:\\x'
162 path = 'c:\\x'
163 m = DummyMagics(ip)
163 m = DummyMagics(ip)
164 opts = m.parse_options('-f %s' % path,'f:')[0]
164 opts = m.parse_options('-f %s' % path,'f:')[0]
165 # argv splitting is os-dependent
165 # argv splitting is os-dependent
166 if os.name == 'posix':
166 if os.name == 'posix':
167 expected = 'c:x'
167 expected = 'c:x'
168 else:
168 else:
169 expected = path
169 expected = path
170 assert opts["f"] == expected
170 assert opts["f"] == expected
171
171
172
172
173 def test_magic_parse_long_options():
173 def test_magic_parse_long_options():
174 """Magic.parse_options can handle --foo=bar long options"""
174 """Magic.parse_options can handle --foo=bar long options"""
175 ip = get_ipython()
175 ip = get_ipython()
176 m = DummyMagics(ip)
176 m = DummyMagics(ip)
177 opts, _ = m.parse_options("--foo --bar=bubble", "a", "foo", "bar=")
177 opts, _ = m.parse_options("--foo --bar=bubble", "a", "foo", "bar=")
178 assert "foo" in opts
178 assert "foo" in opts
179 assert "bar" in opts
179 assert "bar" in opts
180 assert opts["bar"] == "bubble"
180 assert opts["bar"] == "bubble"
181
181
182
182
183 def doctest_hist_f():
183 def doctest_hist_f():
184 """Test %hist -f with temporary filename.
184 """Test %hist -f with temporary filename.
185
185
186 In [9]: import tempfile
186 In [9]: import tempfile
187
187
188 In [10]: tfile = tempfile.mktemp('.py','tmp-ipython-')
188 In [10]: tfile = tempfile.mktemp('.py','tmp-ipython-')
189
189
190 In [11]: %hist -nl -f $tfile 3
190 In [11]: %hist -nl -f $tfile 3
191
191
192 In [13]: import os; os.unlink(tfile)
192 In [13]: import os; os.unlink(tfile)
193 """
193 """
194
194
195
195
196 def doctest_hist_op():
196 def doctest_hist_op():
197 """Test %hist -op
197 """Test %hist -op
198
198
199 In [1]: class b(float):
199 In [1]: class b(float):
200 ...: pass
200 ...: pass
201 ...:
201 ...:
202
202
203 In [2]: class s(object):
203 In [2]: class s(object):
204 ...: def __str__(self):
204 ...: def __str__(self):
205 ...: return 's'
205 ...: return 's'
206 ...:
206 ...:
207
207
208 In [3]:
208 In [3]:
209
209
210 In [4]: class r(b):
210 In [4]: class r(b):
211 ...: def __repr__(self):
211 ...: def __repr__(self):
212 ...: return 'r'
212 ...: return 'r'
213 ...:
213 ...:
214
214
215 In [5]: class sr(s,r): pass
215 In [5]: class sr(s,r): pass
216 ...:
216 ...:
217
217
218 In [6]:
218 In [6]:
219
219
220 In [7]: bb=b()
220 In [7]: bb=b()
221
221
222 In [8]: ss=s()
222 In [8]: ss=s()
223
223
224 In [9]: rr=r()
224 In [9]: rr=r()
225
225
226 In [10]: ssrr=sr()
226 In [10]: ssrr=sr()
227
227
228 In [11]: 4.5
228 In [11]: 4.5
229 Out[11]: 4.5
229 Out[11]: 4.5
230
230
231 In [12]: str(ss)
231 In [12]: str(ss)
232 Out[12]: 's'
232 Out[12]: 's'
233
233
234 In [13]:
234 In [13]:
235
235
236 In [14]: %hist -op
236 In [14]: %hist -op
237 >>> class b:
237 >>> class b:
238 ... pass
238 ... pass
239 ...
239 ...
240 >>> class s(b):
240 >>> class s(b):
241 ... def __str__(self):
241 ... def __str__(self):
242 ... return 's'
242 ... return 's'
243 ...
243 ...
244 >>>
244 >>>
245 >>> class r(b):
245 >>> class r(b):
246 ... def __repr__(self):
246 ... def __repr__(self):
247 ... return 'r'
247 ... return 'r'
248 ...
248 ...
249 >>> class sr(s,r): pass
249 >>> class sr(s,r): pass
250 >>>
250 >>>
251 >>> bb=b()
251 >>> bb=b()
252 >>> ss=s()
252 >>> ss=s()
253 >>> rr=r()
253 >>> rr=r()
254 >>> ssrr=sr()
254 >>> ssrr=sr()
255 >>> 4.5
255 >>> 4.5
256 4.5
256 4.5
257 >>> str(ss)
257 >>> str(ss)
258 's'
258 's'
259 >>>
259 >>>
260 """
260 """
261
261
262 def test_hist_pof():
262 def test_hist_pof():
263 ip = get_ipython()
263 ip = get_ipython()
264 ip.run_cell("1+2", store_history=True)
264 ip.run_cell("1+2", store_history=True)
265 #raise Exception(ip.history_manager.session_number)
265 #raise Exception(ip.history_manager.session_number)
266 #raise Exception(list(ip.history_manager._get_range_session()))
266 #raise Exception(list(ip.history_manager._get_range_session()))
267 with TemporaryDirectory() as td:
267 with TemporaryDirectory() as td:
268 tf = os.path.join(td, 'hist.py')
268 tf = os.path.join(td, 'hist.py')
269 ip.run_line_magic('history', '-pof %s' % tf)
269 ip.run_line_magic('history', '-pof %s' % tf)
270 assert os.path.isfile(tf)
270 assert os.path.isfile(tf)
271
271
272
272
273 def test_macro():
273 def test_macro():
274 ip = get_ipython()
274 ip = get_ipython()
275 ip.history_manager.reset() # Clear any existing history.
275 ip.history_manager.reset() # Clear any existing history.
276 cmds = ["a=1", "def b():\n return a**2", "print(a,b())"]
276 cmds = ["a=1", "def b():\n return a**2", "print(a,b())"]
277 for i, cmd in enumerate(cmds, start=1):
277 for i, cmd in enumerate(cmds, start=1):
278 ip.history_manager.store_inputs(i, cmd)
278 ip.history_manager.store_inputs(i, cmd)
279 ip.magic("macro test 1-3")
279 ip.magic("macro test 1-3")
280 assert ip.user_ns["test"].value == "\n".join(cmds) + "\n"
280 assert ip.user_ns["test"].value == "\n".join(cmds) + "\n"
281
281
282 # List macros
282 # List macros
283 assert "test" in ip.magic("macro")
283 assert "test" in ip.magic("macro")
284
284
285
285
286 def test_macro_run():
286 def test_macro_run():
287 """Test that we can run a multi-line macro successfully."""
287 """Test that we can run a multi-line macro successfully."""
288 ip = get_ipython()
288 ip = get_ipython()
289 ip.history_manager.reset()
289 ip.history_manager.reset()
290 cmds = ["a=10", "a+=1", "print(a)", "%macro test 2-3"]
290 cmds = ["a=10", "a+=1", "print(a)", "%macro test 2-3"]
291 for cmd in cmds:
291 for cmd in cmds:
292 ip.run_cell(cmd, store_history=True)
292 ip.run_cell(cmd, store_history=True)
293 assert ip.user_ns["test"].value == "a+=1\nprint(a)\n"
293 assert ip.user_ns["test"].value == "a+=1\nprint(a)\n"
294 with tt.AssertPrints("12"):
294 with tt.AssertPrints("12"):
295 ip.run_cell("test")
295 ip.run_cell("test")
296 with tt.AssertPrints("13"):
296 with tt.AssertPrints("13"):
297 ip.run_cell("test")
297 ip.run_cell("test")
298
298
299
299
300 def test_magic_magic():
300 def test_magic_magic():
301 """Test %magic"""
301 """Test %magic"""
302 ip = get_ipython()
302 ip = get_ipython()
303 with capture_output() as captured:
303 with capture_output() as captured:
304 ip.magic("magic")
304 ip.magic("magic")
305
305
306 stdout = captured.stdout
306 stdout = captured.stdout
307 assert "%magic" in stdout
307 assert "%magic" in stdout
308 assert "IPython" in stdout
308 assert "IPython" in stdout
309 assert "Available" in stdout
309 assert "Available" in stdout
310
310
311
311
312 @dec.skipif_not_numpy
312 @dec.skipif_not_numpy
313 def test_numpy_reset_array_undec():
313 def test_numpy_reset_array_undec():
314 "Test '%reset array' functionality"
314 "Test '%reset array' functionality"
315 _ip.ex("import numpy as np")
315 _ip.ex("import numpy as np")
316 _ip.ex("a = np.empty(2)")
316 _ip.ex("a = np.empty(2)")
317 assert "a" in _ip.user_ns
317 assert "a" in _ip.user_ns
318 _ip.magic("reset -f array")
318 _ip.magic("reset -f array")
319 assert "a" not in _ip.user_ns
319 assert "a" not in _ip.user_ns
320
320
321
321
322 def test_reset_out():
322 def test_reset_out():
323 "Test '%reset out' magic"
323 "Test '%reset out' magic"
324 _ip.run_cell("parrot = 'dead'", store_history=True)
324 _ip.run_cell("parrot = 'dead'", store_history=True)
325 # test '%reset -f out', make an Out prompt
325 # test '%reset -f out', make an Out prompt
326 _ip.run_cell("parrot", store_history=True)
326 _ip.run_cell("parrot", store_history=True)
327 assert "dead" in [_ip.user_ns[x] for x in ("_", "__", "___")]
327 assert "dead" in [_ip.user_ns[x] for x in ("_", "__", "___")]
328 _ip.magic("reset -f out")
328 _ip.magic("reset -f out")
329 assert "dead" not in [_ip.user_ns[x] for x in ("_", "__", "___")]
329 assert "dead" not in [_ip.user_ns[x] for x in ("_", "__", "___")]
330 assert len(_ip.user_ns["Out"]) == 0
330 assert len(_ip.user_ns["Out"]) == 0
331
331
332
332
333 def test_reset_in():
333 def test_reset_in():
334 "Test '%reset in' magic"
334 "Test '%reset in' magic"
335 # test '%reset -f in'
335 # test '%reset -f in'
336 _ip.run_cell("parrot", store_history=True)
336 _ip.run_cell("parrot", store_history=True)
337 assert "parrot" in [_ip.user_ns[x] for x in ("_i", "_ii", "_iii")]
337 assert "parrot" in [_ip.user_ns[x] for x in ("_i", "_ii", "_iii")]
338 _ip.magic("%reset -f in")
338 _ip.magic("%reset -f in")
339 assert "parrot" not in [_ip.user_ns[x] for x in ("_i", "_ii", "_iii")]
339 assert "parrot" not in [_ip.user_ns[x] for x in ("_i", "_ii", "_iii")]
340 assert len(set(_ip.user_ns["In"])) == 1
340 assert len(set(_ip.user_ns["In"])) == 1
341
341
342
342
343 def test_reset_dhist():
343 def test_reset_dhist():
344 "Test '%reset dhist' magic"
344 "Test '%reset dhist' magic"
345 _ip.run_cell("tmp = [d for d in _dh]") # copy before clearing
345 _ip.run_cell("tmp = [d for d in _dh]") # copy before clearing
346 _ip.magic("cd " + os.path.dirname(pytest.__file__))
346 _ip.magic("cd " + os.path.dirname(pytest.__file__))
347 _ip.magic("cd -")
347 _ip.magic("cd -")
348 assert len(_ip.user_ns["_dh"]) > 0
348 assert len(_ip.user_ns["_dh"]) > 0
349 _ip.magic("reset -f dhist")
349 _ip.magic("reset -f dhist")
350 assert len(_ip.user_ns["_dh"]) == 0
350 assert len(_ip.user_ns["_dh"]) == 0
351 _ip.run_cell("_dh = [d for d in tmp]") # restore
351 _ip.run_cell("_dh = [d for d in tmp]") # restore
352
352
353
353
354 def test_reset_in_length():
354 def test_reset_in_length():
355 "Test that '%reset in' preserves In[] length"
355 "Test that '%reset in' preserves In[] length"
356 _ip.run_cell("print 'foo'")
356 _ip.run_cell("print 'foo'")
357 _ip.run_cell("reset -f in")
357 _ip.run_cell("reset -f in")
358 assert len(_ip.user_ns["In"]) == _ip.displayhook.prompt_count + 1
358 assert len(_ip.user_ns["In"]) == _ip.displayhook.prompt_count + 1
359
359
360
360
361 class TestResetErrors(TestCase):
361 class TestResetErrors(TestCase):
362
362
363 def test_reset_redefine(self):
363 def test_reset_redefine(self):
364
364
365 @magics_class
365 @magics_class
366 class KernelMagics(Magics):
366 class KernelMagics(Magics):
367 @line_magic
367 @line_magic
368 def less(self, shell): pass
368 def less(self, shell): pass
369
369
370 _ip.register_magics(KernelMagics)
370 _ip.register_magics(KernelMagics)
371
371
372 with self.assertLogs() as cm:
372 with self.assertLogs() as cm:
373 # hack, we want to just capture logs, but assertLogs fails if not
373 # hack, we want to just capture logs, but assertLogs fails if not
374 # logs get produce.
374 # logs get produce.
375 # so log one things we ignore.
375 # so log one things we ignore.
376 import logging as log_mod
376 import logging as log_mod
377 log = log_mod.getLogger()
377 log = log_mod.getLogger()
378 log.info('Nothing')
378 log.info('Nothing')
379 # end hack.
379 # end hack.
380 _ip.run_cell("reset -f")
380 _ip.run_cell("reset -f")
381
381
382 assert len(cm.output) == 1
382 assert len(cm.output) == 1
383 for out in cm.output:
383 for out in cm.output:
384 assert "Invalid alias" not in out
384 assert "Invalid alias" not in out
385
385
386 def test_tb_syntaxerror():
386 def test_tb_syntaxerror():
387 """test %tb after a SyntaxError"""
387 """test %tb after a SyntaxError"""
388 ip = get_ipython()
388 ip = get_ipython()
389 ip.run_cell("for")
389 ip.run_cell("for")
390
390
391 # trap and validate stdout
391 # trap and validate stdout
392 save_stdout = sys.stdout
392 save_stdout = sys.stdout
393 try:
393 try:
394 sys.stdout = StringIO()
394 sys.stdout = StringIO()
395 ip.run_cell("%tb")
395 ip.run_cell("%tb")
396 out = sys.stdout.getvalue()
396 out = sys.stdout.getvalue()
397 finally:
397 finally:
398 sys.stdout = save_stdout
398 sys.stdout = save_stdout
399 # trim output, and only check the last line
399 # trim output, and only check the last line
400 last_line = out.rstrip().splitlines()[-1].strip()
400 last_line = out.rstrip().splitlines()[-1].strip()
401 assert last_line == "SyntaxError: invalid syntax"
401 assert last_line == "SyntaxError: invalid syntax"
402
402
403
403
404 def test_time():
404 def test_time():
405 ip = get_ipython()
405 ip = get_ipython()
406
406
407 with tt.AssertPrints("Wall time: "):
407 with tt.AssertPrints("Wall time: "):
408 ip.run_cell("%time None")
408 ip.run_cell("%time None")
409
409
410 ip.run_cell("def f(kmjy):\n"
410 ip.run_cell("def f(kmjy):\n"
411 " %time print (2*kmjy)")
411 " %time print (2*kmjy)")
412
412
413 with tt.AssertPrints("Wall time: "):
413 with tt.AssertPrints("Wall time: "):
414 with tt.AssertPrints("hihi", suppress=False):
414 with tt.AssertPrints("hihi", suppress=False):
415 ip.run_cell("f('hi')")
415 ip.run_cell("f('hi')")
416
416
417 def test_time_last_not_expression():
417 def test_time_last_not_expression():
418 ip.run_cell("%%time\n"
418 ip.run_cell("%%time\n"
419 "var_1 = 1\n"
419 "var_1 = 1\n"
420 "var_2 = 2\n")
420 "var_2 = 2\n")
421 assert ip.user_ns['var_1'] == 1
421 assert ip.user_ns['var_1'] == 1
422 del ip.user_ns['var_1']
422 del ip.user_ns['var_1']
423 assert ip.user_ns['var_2'] == 2
423 assert ip.user_ns['var_2'] == 2
424 del ip.user_ns['var_2']
424 del ip.user_ns['var_2']
425
425
426
426
427 @dec.skip_win32
427 @dec.skip_win32
428 def test_time2():
428 def test_time2():
429 ip = get_ipython()
429 ip = get_ipython()
430
430
431 with tt.AssertPrints("CPU times: user "):
431 with tt.AssertPrints("CPU times: user "):
432 ip.run_cell("%time None")
432 ip.run_cell("%time None")
433
433
434 def test_time3():
434 def test_time3():
435 """Erroneous magic function calls, issue gh-3334"""
435 """Erroneous magic function calls, issue gh-3334"""
436 ip = get_ipython()
436 ip = get_ipython()
437 ip.user_ns.pop('run', None)
437 ip.user_ns.pop('run', None)
438
438
439 with tt.AssertNotPrints("not found", channel='stderr'):
439 with tt.AssertNotPrints("not found", channel='stderr'):
440 ip.run_cell("%%time\n"
440 ip.run_cell("%%time\n"
441 "run = 0\n"
441 "run = 0\n"
442 "run += 1")
442 "run += 1")
443
443
444 def test_multiline_time():
444 def test_multiline_time():
445 """Make sure last statement from time return a value."""
445 """Make sure last statement from time return a value."""
446 ip = get_ipython()
446 ip = get_ipython()
447 ip.user_ns.pop('run', None)
447 ip.user_ns.pop('run', None)
448
448
449 ip.run_cell(dedent("""\
449 ip.run_cell(dedent("""\
450 %%time
450 %%time
451 a = "ho"
451 a = "ho"
452 b = "hey"
452 b = "hey"
453 a+b
453 a+b
454 """
454 """
455 )
455 )
456 )
456 )
457 assert ip.user_ns_hidden["_"] == "hohey"
457 assert ip.user_ns_hidden["_"] == "hohey"
458
458
459
459
460 def test_time_local_ns():
460 def test_time_local_ns():
461 """
461 """
462 Test that local_ns is actually global_ns when running a cell magic
462 Test that local_ns is actually global_ns when running a cell magic
463 """
463 """
464 ip = get_ipython()
464 ip = get_ipython()
465 ip.run_cell("%%time\n" "myvar = 1")
465 ip.run_cell("%%time\n" "myvar = 1")
466 assert ip.user_ns["myvar"] == 1
466 assert ip.user_ns["myvar"] == 1
467 del ip.user_ns["myvar"]
467 del ip.user_ns["myvar"]
468
468
469
469
470 def test_doctest_mode():
470 def test_doctest_mode():
471 "Toggle doctest_mode twice, it should be a no-op and run without error"
471 "Toggle doctest_mode twice, it should be a no-op and run without error"
472 _ip.magic('doctest_mode')
472 _ip.magic('doctest_mode')
473 _ip.magic('doctest_mode')
473 _ip.magic('doctest_mode')
474
474
475
475
476 def test_parse_options():
476 def test_parse_options():
477 """Tests for basic options parsing in magics."""
477 """Tests for basic options parsing in magics."""
478 # These are only the most minimal of tests, more should be added later. At
478 # These are only the most minimal of tests, more should be added later. At
479 # the very least we check that basic text/unicode calls work OK.
479 # the very least we check that basic text/unicode calls work OK.
480 m = DummyMagics(_ip)
480 m = DummyMagics(_ip)
481 assert m.parse_options("foo", "")[1] == "foo"
481 assert m.parse_options("foo", "")[1] == "foo"
482 assert m.parse_options("foo", "")[1] == "foo"
482 assert m.parse_options("foo", "")[1] == "foo"
483
483
484
484
485 def test_parse_options_preserve_non_option_string():
485 def test_parse_options_preserve_non_option_string():
486 """Test to assert preservation of non-option part of magic-block, while parsing magic options."""
486 """Test to assert preservation of non-option part of magic-block, while parsing magic options."""
487 m = DummyMagics(_ip)
487 m = DummyMagics(_ip)
488 opts, stmt = m.parse_options(
488 opts, stmt = m.parse_options(
489 " -n1 -r 13 _ = 314 + foo", "n:r:", preserve_non_opts=True
489 " -n1 -r 13 _ = 314 + foo", "n:r:", preserve_non_opts=True
490 )
490 )
491 assert opts == {"n": "1", "r": "13"}
491 assert opts == {"n": "1", "r": "13"}
492 assert stmt == "_ = 314 + foo"
492 assert stmt == "_ = 314 + foo"
493
493
494
494
495 def test_run_magic_preserve_code_block():
495 def test_run_magic_preserve_code_block():
496 """Test to assert preservation of non-option part of magic-block, while running magic."""
496 """Test to assert preservation of non-option part of magic-block, while running magic."""
497 _ip.user_ns["spaces"] = []
497 _ip.user_ns["spaces"] = []
498 _ip.magic("timeit -n1 -r1 spaces.append([s.count(' ') for s in ['document']])")
498 _ip.magic("timeit -n1 -r1 spaces.append([s.count(' ') for s in ['document']])")
499 assert _ip.user_ns["spaces"] == [[0]]
499 assert _ip.user_ns["spaces"] == [[0]]
500
500
501
501
502 def test_dirops():
502 def test_dirops():
503 """Test various directory handling operations."""
503 """Test various directory handling operations."""
504 # curpath = lambda :os.path.splitdrive(os.getcwd())[1].replace('\\','/')
504 # curpath = lambda :os.path.splitdrive(os.getcwd())[1].replace('\\','/')
505 curpath = os.getcwd
505 curpath = os.getcwd
506 startdir = os.getcwd()
506 startdir = os.getcwd()
507 ipdir = os.path.realpath(_ip.ipython_dir)
507 ipdir = os.path.realpath(_ip.ipython_dir)
508 try:
508 try:
509 _ip.magic('cd "%s"' % ipdir)
509 _ip.magic('cd "%s"' % ipdir)
510 assert curpath() == ipdir
510 assert curpath() == ipdir
511 _ip.magic('cd -')
511 _ip.magic('cd -')
512 assert curpath() == startdir
512 assert curpath() == startdir
513 _ip.magic('pushd "%s"' % ipdir)
513 _ip.magic('pushd "%s"' % ipdir)
514 assert curpath() == ipdir
514 assert curpath() == ipdir
515 _ip.magic('popd')
515 _ip.magic('popd')
516 assert curpath() == startdir
516 assert curpath() == startdir
517 finally:
517 finally:
518 os.chdir(startdir)
518 os.chdir(startdir)
519
519
520
520
521 def test_cd_force_quiet():
521 def test_cd_force_quiet():
522 """Test OSMagics.cd_force_quiet option"""
522 """Test OSMagics.cd_force_quiet option"""
523 _ip.config.OSMagics.cd_force_quiet = True
523 _ip.config.OSMagics.cd_force_quiet = True
524 osmagics = osm.OSMagics(shell=_ip)
524 osmagics = osm.OSMagics(shell=_ip)
525
525
526 startdir = os.getcwd()
526 startdir = os.getcwd()
527 ipdir = os.path.realpath(_ip.ipython_dir)
527 ipdir = os.path.realpath(_ip.ipython_dir)
528
528
529 try:
529 try:
530 with tt.AssertNotPrints(ipdir):
530 with tt.AssertNotPrints(ipdir):
531 osmagics.cd('"%s"' % ipdir)
531 osmagics.cd('"%s"' % ipdir)
532 with tt.AssertNotPrints(startdir):
532 with tt.AssertNotPrints(startdir):
533 osmagics.cd('-')
533 osmagics.cd('-')
534 finally:
534 finally:
535 os.chdir(startdir)
535 os.chdir(startdir)
536
536
537
537
538 def test_xmode():
538 def test_xmode():
539 # Calling xmode three times should be a no-op
539 # Calling xmode three times should be a no-op
540 xmode = _ip.InteractiveTB.mode
540 xmode = _ip.InteractiveTB.mode
541 for i in range(4):
541 for i in range(4):
542 _ip.magic("xmode")
542 _ip.magic("xmode")
543 assert _ip.InteractiveTB.mode == xmode
543 assert _ip.InteractiveTB.mode == xmode
544
544
545 def test_reset_hard():
545 def test_reset_hard():
546 monitor = []
546 monitor = []
547 class A(object):
547 class A(object):
548 def __del__(self):
548 def __del__(self):
549 monitor.append(1)
549 monitor.append(1)
550 def __repr__(self):
550 def __repr__(self):
551 return "<A instance>"
551 return "<A instance>"
552
552
553 _ip.user_ns["a"] = A()
553 _ip.user_ns["a"] = A()
554 _ip.run_cell("a")
554 _ip.run_cell("a")
555
555
556 assert monitor == []
556 assert monitor == []
557 _ip.magic("reset -f")
557 _ip.magic("reset -f")
558 assert monitor == [1]
558 assert monitor == [1]
559
559
560 class TestXdel(tt.TempFileMixin):
560 class TestXdel(tt.TempFileMixin):
561 def test_xdel(self):
561 def test_xdel(self):
562 """Test that references from %run are cleared by xdel."""
562 """Test that references from %run are cleared by xdel."""
563 src = ("class A(object):\n"
563 src = ("class A(object):\n"
564 " monitor = []\n"
564 " monitor = []\n"
565 " def __del__(self):\n"
565 " def __del__(self):\n"
566 " self.monitor.append(1)\n"
566 " self.monitor.append(1)\n"
567 "a = A()\n")
567 "a = A()\n")
568 self.mktmp(src)
568 self.mktmp(src)
569 # %run creates some hidden references...
569 # %run creates some hidden references...
570 _ip.magic("run %s" % self.fname)
570 _ip.magic("run %s" % self.fname)
571 # ... as does the displayhook.
571 # ... as does the displayhook.
572 _ip.run_cell("a")
572 _ip.run_cell("a")
573
573
574 monitor = _ip.user_ns["A"].monitor
574 monitor = _ip.user_ns["A"].monitor
575 assert monitor == []
575 assert monitor == []
576
576
577 _ip.magic("xdel a")
577 _ip.magic("xdel a")
578
578
579 # Check that a's __del__ method has been called.
579 # Check that a's __del__ method has been called.
580 gc.collect(0)
580 gc.collect(0)
581 assert monitor == [1]
581 assert monitor == [1]
582
582
583 def doctest_who():
583 def doctest_who():
584 """doctest for %who
584 """doctest for %who
585
585
586 In [1]: %reset -sf
586 In [1]: %reset -sf
587
587
588 In [2]: alpha = 123
588 In [2]: alpha = 123
589
589
590 In [3]: beta = 'beta'
590 In [3]: beta = 'beta'
591
591
592 In [4]: %who int
592 In [4]: %who int
593 alpha
593 alpha
594
594
595 In [5]: %who str
595 In [5]: %who str
596 beta
596 beta
597
597
598 In [6]: %whos
598 In [6]: %whos
599 Variable Type Data/Info
599 Variable Type Data/Info
600 ----------------------------
600 ----------------------------
601 alpha int 123
601 alpha int 123
602 beta str beta
602 beta str beta
603
603
604 In [7]: %who_ls
604 In [7]: %who_ls
605 Out[7]: ['alpha', 'beta']
605 Out[7]: ['alpha', 'beta']
606 """
606 """
607
607
608 def test_whos():
608 def test_whos():
609 """Check that whos is protected against objects where repr() fails."""
609 """Check that whos is protected against objects where repr() fails."""
610 class A(object):
610 class A(object):
611 def __repr__(self):
611 def __repr__(self):
612 raise Exception()
612 raise Exception()
613 _ip.user_ns['a'] = A()
613 _ip.user_ns['a'] = A()
614 _ip.magic("whos")
614 _ip.magic("whos")
615
615
616 def doctest_precision():
616 def doctest_precision():
617 """doctest for %precision
617 """doctest for %precision
618
618
619 In [1]: f = get_ipython().display_formatter.formatters['text/plain']
619 In [1]: f = get_ipython().display_formatter.formatters['text/plain']
620
620
621 In [2]: %precision 5
621 In [2]: %precision 5
622 Out[2]: '%.5f'
622 Out[2]: '%.5f'
623
623
624 In [3]: f.float_format
624 In [3]: f.float_format
625 Out[3]: '%.5f'
625 Out[3]: '%.5f'
626
626
627 In [4]: %precision %e
627 In [4]: %precision %e
628 Out[4]: '%e'
628 Out[4]: '%e'
629
629
630 In [5]: f(3.1415927)
630 In [5]: f(3.1415927)
631 Out[5]: '3.141593e+00'
631 Out[5]: '3.141593e+00'
632 """
632 """
633
633
634 def test_debug_magic():
634 def test_debug_magic():
635 """Test debugging a small code with %debug
635 """Test debugging a small code with %debug
636
636
637 In [1]: with PdbTestInput(['c']):
637 In [1]: with PdbTestInput(['c']):
638 ...: %debug print("a b") #doctest: +ELLIPSIS
638 ...: %debug print("a b") #doctest: +ELLIPSIS
639 ...:
639 ...:
640 ...
640 ...
641 ipdb> c
641 ipdb> c
642 a b
642 a b
643 In [2]:
643 In [2]:
644 """
644 """
645
645
646 def test_psearch():
646 def test_psearch():
647 with tt.AssertPrints("dict.fromkeys"):
647 with tt.AssertPrints("dict.fromkeys"):
648 _ip.run_cell("dict.fr*?")
648 _ip.run_cell("dict.fr*?")
649 with tt.AssertPrints("Ο€.is_integer"):
649 with tt.AssertPrints("Ο€.is_integer"):
650 _ip.run_cell("Ο€ = 3.14;\nΟ€.is_integ*?")
650 _ip.run_cell("Ο€ = 3.14;\nΟ€.is_integ*?")
651
651
652 def test_timeit_shlex():
652 def test_timeit_shlex():
653 """test shlex issues with timeit (#1109)"""
653 """test shlex issues with timeit (#1109)"""
654 _ip.ex("def f(*a,**kw): pass")
654 _ip.ex("def f(*a,**kw): pass")
655 _ip.magic('timeit -n1 "this is a bug".count(" ")')
655 _ip.magic('timeit -n1 "this is a bug".count(" ")')
656 _ip.magic('timeit -r1 -n1 f(" ", 1)')
656 _ip.magic('timeit -r1 -n1 f(" ", 1)')
657 _ip.magic('timeit -r1 -n1 f(" ", 1, " ", 2, " ")')
657 _ip.magic('timeit -r1 -n1 f(" ", 1, " ", 2, " ")')
658 _ip.magic('timeit -r1 -n1 ("a " + "b")')
658 _ip.magic('timeit -r1 -n1 ("a " + "b")')
659 _ip.magic('timeit -r1 -n1 f("a " + "b")')
659 _ip.magic('timeit -r1 -n1 f("a " + "b")')
660 _ip.magic('timeit -r1 -n1 f("a " + "b ")')
660 _ip.magic('timeit -r1 -n1 f("a " + "b ")')
661
661
662
662
663 def test_timeit_special_syntax():
663 def test_timeit_special_syntax():
664 "Test %%timeit with IPython special syntax"
664 "Test %%timeit with IPython special syntax"
665 @register_line_magic
665 @register_line_magic
666 def lmagic(line):
666 def lmagic(line):
667 ip = get_ipython()
667 ip = get_ipython()
668 ip.user_ns['lmagic_out'] = line
668 ip.user_ns['lmagic_out'] = line
669
669
670 # line mode test
670 # line mode test
671 _ip.run_line_magic("timeit", "-n1 -r1 %lmagic my line")
671 _ip.run_line_magic("timeit", "-n1 -r1 %lmagic my line")
672 assert _ip.user_ns["lmagic_out"] == "my line"
672 assert _ip.user_ns["lmagic_out"] == "my line"
673 # cell mode test
673 # cell mode test
674 _ip.run_cell_magic("timeit", "-n1 -r1", "%lmagic my line2")
674 _ip.run_cell_magic("timeit", "-n1 -r1", "%lmagic my line2")
675 assert _ip.user_ns["lmagic_out"] == "my line2"
675 assert _ip.user_ns["lmagic_out"] == "my line2"
676
676
677
677
678 def test_timeit_return():
678 def test_timeit_return():
679 """
679 """
680 test whether timeit -o return object
680 test whether timeit -o return object
681 """
681 """
682
682
683 res = _ip.run_line_magic('timeit','-n10 -r10 -o 1')
683 res = _ip.run_line_magic('timeit','-n10 -r10 -o 1')
684 assert(res is not None)
684 assert(res is not None)
685
685
686 def test_timeit_quiet():
686 def test_timeit_quiet():
687 """
687 """
688 test quiet option of timeit magic
688 test quiet option of timeit magic
689 """
689 """
690 with tt.AssertNotPrints("loops"):
690 with tt.AssertNotPrints("loops"):
691 _ip.run_cell("%timeit -n1 -r1 -q 1")
691 _ip.run_cell("%timeit -n1 -r1 -q 1")
692
692
693 def test_timeit_return_quiet():
693 def test_timeit_return_quiet():
694 with tt.AssertNotPrints("loops"):
694 with tt.AssertNotPrints("loops"):
695 res = _ip.run_line_magic('timeit', '-n1 -r1 -q -o 1')
695 res = _ip.run_line_magic('timeit', '-n1 -r1 -q -o 1')
696 assert (res is not None)
696 assert (res is not None)
697
697
698 def test_timeit_invalid_return():
698 def test_timeit_invalid_return():
699 with pytest.raises(SyntaxError):
699 with pytest.raises(SyntaxError):
700 _ip.run_line_magic('timeit', 'return')
700 _ip.run_line_magic('timeit', 'return')
701
701
702 @dec.skipif(execution.profile is None)
702 @dec.skipif(execution.profile is None)
703 def test_prun_special_syntax():
703 def test_prun_special_syntax():
704 "Test %%prun with IPython special syntax"
704 "Test %%prun with IPython special syntax"
705 @register_line_magic
705 @register_line_magic
706 def lmagic(line):
706 def lmagic(line):
707 ip = get_ipython()
707 ip = get_ipython()
708 ip.user_ns['lmagic_out'] = line
708 ip.user_ns['lmagic_out'] = line
709
709
710 # line mode test
710 # line mode test
711 _ip.run_line_magic("prun", "-q %lmagic my line")
711 _ip.run_line_magic("prun", "-q %lmagic my line")
712 assert _ip.user_ns["lmagic_out"] == "my line"
712 assert _ip.user_ns["lmagic_out"] == "my line"
713 # cell mode test
713 # cell mode test
714 _ip.run_cell_magic("prun", "-q", "%lmagic my line2")
714 _ip.run_cell_magic("prun", "-q", "%lmagic my line2")
715 assert _ip.user_ns["lmagic_out"] == "my line2"
715 assert _ip.user_ns["lmagic_out"] == "my line2"
716
716
717
717
718 @dec.skipif(execution.profile is None)
718 @dec.skipif(execution.profile is None)
719 def test_prun_quotes():
719 def test_prun_quotes():
720 "Test that prun does not clobber string escapes (GH #1302)"
720 "Test that prun does not clobber string escapes (GH #1302)"
721 _ip.magic(r"prun -q x = '\t'")
721 _ip.magic(r"prun -q x = '\t'")
722 assert _ip.user_ns["x"] == "\t"
722 assert _ip.user_ns["x"] == "\t"
723
723
724
724
725 def test_extension():
725 def test_extension():
726 # Debugging information for failures of this test
726 # Debugging information for failures of this test
727 print('sys.path:')
727 print('sys.path:')
728 for p in sys.path:
728 for p in sys.path:
729 print(' ', p)
729 print(' ', p)
730 print('CWD', os.getcwd())
730 print('CWD', os.getcwd())
731
731
732 pytest.raises(ImportError, _ip.magic, "load_ext daft_extension")
732 pytest.raises(ImportError, _ip.magic, "load_ext daft_extension")
733 daft_path = os.path.join(os.path.dirname(__file__), "daft_extension")
733 daft_path = os.path.join(os.path.dirname(__file__), "daft_extension")
734 sys.path.insert(0, daft_path)
734 sys.path.insert(0, daft_path)
735 try:
735 try:
736 _ip.user_ns.pop('arq', None)
736 _ip.user_ns.pop('arq', None)
737 invalidate_caches() # Clear import caches
737 invalidate_caches() # Clear import caches
738 _ip.magic("load_ext daft_extension")
738 _ip.magic("load_ext daft_extension")
739 assert _ip.user_ns["arq"] == 185
739 assert _ip.user_ns["arq"] == 185
740 _ip.magic("unload_ext daft_extension")
740 _ip.magic("unload_ext daft_extension")
741 assert 'arq' not in _ip.user_ns
741 assert 'arq' not in _ip.user_ns
742 finally:
742 finally:
743 sys.path.remove(daft_path)
743 sys.path.remove(daft_path)
744
744
745
745
746 def test_notebook_export_json():
746 def test_notebook_export_json():
747 pytest.importorskip("nbformat")
747 pytest.importorskip("nbformat")
748 _ip = get_ipython()
748 _ip = get_ipython()
749 _ip.history_manager.reset() # Clear any existing history.
749 _ip.history_manager.reset() # Clear any existing history.
750 cmds = ["a=1", "def b():\n return a**2", "print('noΓ«l, Γ©tΓ©', b())"]
750 cmds = ["a=1", "def b():\n return a**2", "print('noΓ«l, Γ©tΓ©', b())"]
751 for i, cmd in enumerate(cmds, start=1):
751 for i, cmd in enumerate(cmds, start=1):
752 _ip.history_manager.store_inputs(i, cmd)
752 _ip.history_manager.store_inputs(i, cmd)
753 with TemporaryDirectory() as td:
753 with TemporaryDirectory() as td:
754 outfile = os.path.join(td, "nb.ipynb")
754 outfile = os.path.join(td, "nb.ipynb")
755 _ip.magic("notebook %s" % outfile)
755 _ip.magic("notebook %s" % outfile)
756
756
757
757
758 class TestEnv(TestCase):
758 class TestEnv(TestCase):
759
759
760 def test_env(self):
760 def test_env(self):
761 env = _ip.magic("env")
761 env = _ip.magic("env")
762 self.assertTrue(isinstance(env, dict))
762 self.assertTrue(isinstance(env, dict))
763
763
764 def test_env_secret(self):
764 def test_env_secret(self):
765 env = _ip.magic("env")
765 env = _ip.magic("env")
766 hidden = "<hidden>"
766 hidden = "<hidden>"
767 with mock.patch.dict(
767 with mock.patch.dict(
768 os.environ,
768 os.environ,
769 {
769 {
770 "API_KEY": "abc123",
770 "API_KEY": "abc123",
771 "SECRET_THING": "ssshhh",
771 "SECRET_THING": "ssshhh",
772 "JUPYTER_TOKEN": "",
772 "JUPYTER_TOKEN": "",
773 "VAR": "abc"
773 "VAR": "abc"
774 }
774 }
775 ):
775 ):
776 env = _ip.magic("env")
776 env = _ip.magic("env")
777 assert env["API_KEY"] == hidden
777 assert env["API_KEY"] == hidden
778 assert env["SECRET_THING"] == hidden
778 assert env["SECRET_THING"] == hidden
779 assert env["JUPYTER_TOKEN"] == hidden
779 assert env["JUPYTER_TOKEN"] == hidden
780 assert env["VAR"] == "abc"
780 assert env["VAR"] == "abc"
781
781
782 def test_env_get_set_simple(self):
782 def test_env_get_set_simple(self):
783 env = _ip.magic("env var val1")
783 env = _ip.magic("env var val1")
784 self.assertEqual(env, None)
784 self.assertEqual(env, None)
785 self.assertEqual(os.environ['var'], 'val1')
785 self.assertEqual(os.environ['var'], 'val1')
786 self.assertEqual(_ip.magic("env var"), 'val1')
786 self.assertEqual(_ip.magic("env var"), 'val1')
787 env = _ip.magic("env var=val2")
787 env = _ip.magic("env var=val2")
788 self.assertEqual(env, None)
788 self.assertEqual(env, None)
789 self.assertEqual(os.environ['var'], 'val2')
789 self.assertEqual(os.environ['var'], 'val2')
790
790
791 def test_env_get_set_complex(self):
791 def test_env_get_set_complex(self):
792 env = _ip.magic("env var 'val1 '' 'val2")
792 env = _ip.magic("env var 'val1 '' 'val2")
793 self.assertEqual(env, None)
793 self.assertEqual(env, None)
794 self.assertEqual(os.environ['var'], "'val1 '' 'val2")
794 self.assertEqual(os.environ['var'], "'val1 '' 'val2")
795 self.assertEqual(_ip.magic("env var"), "'val1 '' 'val2")
795 self.assertEqual(_ip.magic("env var"), "'val1 '' 'val2")
796 env = _ip.magic('env var=val2 val3="val4')
796 env = _ip.magic('env var=val2 val3="val4')
797 self.assertEqual(env, None)
797 self.assertEqual(env, None)
798 self.assertEqual(os.environ['var'], 'val2 val3="val4')
798 self.assertEqual(os.environ['var'], 'val2 val3="val4')
799
799
800 def test_env_set_bad_input(self):
800 def test_env_set_bad_input(self):
801 self.assertRaises(UsageError, lambda: _ip.magic("set_env var"))
801 self.assertRaises(UsageError, lambda: _ip.magic("set_env var"))
802
802
803 def test_env_set_whitespace(self):
803 def test_env_set_whitespace(self):
804 self.assertRaises(UsageError, lambda: _ip.magic("env var A=B"))
804 self.assertRaises(UsageError, lambda: _ip.magic("env var A=B"))
805
805
806
806
807 class CellMagicTestCase(TestCase):
807 class CellMagicTestCase(TestCase):
808
808
809 def check_ident(self, magic):
809 def check_ident(self, magic):
810 # Manually called, we get the result
810 # Manually called, we get the result
811 out = _ip.run_cell_magic(magic, "a", "b")
811 out = _ip.run_cell_magic(magic, "a", "b")
812 assert out == ("a", "b")
812 assert out == ("a", "b")
813 # Via run_cell, it goes into the user's namespace via displayhook
813 # Via run_cell, it goes into the user's namespace via displayhook
814 _ip.run_cell("%%" + magic + " c\nd\n")
814 _ip.run_cell("%%" + magic + " c\nd\n")
815 assert _ip.user_ns["_"] == ("c", "d\n")
815 assert _ip.user_ns["_"] == ("c", "d\n")
816
816
817 def test_cell_magic_func_deco(self):
817 def test_cell_magic_func_deco(self):
818 "Cell magic using simple decorator"
818 "Cell magic using simple decorator"
819 @register_cell_magic
819 @register_cell_magic
820 def cellm(line, cell):
820 def cellm(line, cell):
821 return line, cell
821 return line, cell
822
822
823 self.check_ident('cellm')
823 self.check_ident('cellm')
824
824
825 def test_cell_magic_reg(self):
825 def test_cell_magic_reg(self):
826 "Cell magic manually registered"
826 "Cell magic manually registered"
827 def cellm(line, cell):
827 def cellm(line, cell):
828 return line, cell
828 return line, cell
829
829
830 _ip.register_magic_function(cellm, 'cell', 'cellm2')
830 _ip.register_magic_function(cellm, 'cell', 'cellm2')
831 self.check_ident('cellm2')
831 self.check_ident('cellm2')
832
832
833 def test_cell_magic_class(self):
833 def test_cell_magic_class(self):
834 "Cell magics declared via a class"
834 "Cell magics declared via a class"
835 @magics_class
835 @magics_class
836 class MyMagics(Magics):
836 class MyMagics(Magics):
837
837
838 @cell_magic
838 @cell_magic
839 def cellm3(self, line, cell):
839 def cellm3(self, line, cell):
840 return line, cell
840 return line, cell
841
841
842 _ip.register_magics(MyMagics)
842 _ip.register_magics(MyMagics)
843 self.check_ident('cellm3')
843 self.check_ident('cellm3')
844
844
845 def test_cell_magic_class2(self):
845 def test_cell_magic_class2(self):
846 "Cell magics declared via a class, #2"
846 "Cell magics declared via a class, #2"
847 @magics_class
847 @magics_class
848 class MyMagics2(Magics):
848 class MyMagics2(Magics):
849
849
850 @cell_magic('cellm4')
850 @cell_magic('cellm4')
851 def cellm33(self, line, cell):
851 def cellm33(self, line, cell):
852 return line, cell
852 return line, cell
853
853
854 _ip.register_magics(MyMagics2)
854 _ip.register_magics(MyMagics2)
855 self.check_ident('cellm4')
855 self.check_ident('cellm4')
856 # Check that nothing is registered as 'cellm33'
856 # Check that nothing is registered as 'cellm33'
857 c33 = _ip.find_cell_magic('cellm33')
857 c33 = _ip.find_cell_magic('cellm33')
858 assert c33 == None
858 assert c33 == None
859
859
860 def test_file():
860 def test_file():
861 """Basic %%writefile"""
861 """Basic %%writefile"""
862 ip = get_ipython()
862 ip = get_ipython()
863 with TemporaryDirectory() as td:
863 with TemporaryDirectory() as td:
864 fname = os.path.join(td, 'file1')
864 fname = os.path.join(td, 'file1')
865 ip.run_cell_magic("writefile", fname, u'\n'.join([
865 ip.run_cell_magic("writefile", fname, u'\n'.join([
866 'line1',
866 'line1',
867 'line2',
867 'line2',
868 ]))
868 ]))
869 s = Path(fname).read_text()
869 s = Path(fname).read_text()
870 assert "line1\n" in s
870 assert "line1\n" in s
871 assert "line2" in s
871 assert "line2" in s
872
872
873
873
874 @dec.skip_win32
874 @dec.skip_win32
875 def test_file_single_quote():
875 def test_file_single_quote():
876 """Basic %%writefile with embedded single quotes"""
876 """Basic %%writefile with embedded single quotes"""
877 ip = get_ipython()
877 ip = get_ipython()
878 with TemporaryDirectory() as td:
878 with TemporaryDirectory() as td:
879 fname = os.path.join(td, '\'file1\'')
879 fname = os.path.join(td, '\'file1\'')
880 ip.run_cell_magic("writefile", fname, u'\n'.join([
880 ip.run_cell_magic("writefile", fname, u'\n'.join([
881 'line1',
881 'line1',
882 'line2',
882 'line2',
883 ]))
883 ]))
884 s = Path(fname).read_text()
884 s = Path(fname).read_text()
885 assert "line1\n" in s
885 assert "line1\n" in s
886 assert "line2" in s
886 assert "line2" in s
887
887
888
888
889 @dec.skip_win32
889 @dec.skip_win32
890 def test_file_double_quote():
890 def test_file_double_quote():
891 """Basic %%writefile with embedded double quotes"""
891 """Basic %%writefile with embedded double quotes"""
892 ip = get_ipython()
892 ip = get_ipython()
893 with TemporaryDirectory() as td:
893 with TemporaryDirectory() as td:
894 fname = os.path.join(td, '"file1"')
894 fname = os.path.join(td, '"file1"')
895 ip.run_cell_magic("writefile", fname, u'\n'.join([
895 ip.run_cell_magic("writefile", fname, u'\n'.join([
896 'line1',
896 'line1',
897 'line2',
897 'line2',
898 ]))
898 ]))
899 s = Path(fname).read_text()
899 s = Path(fname).read_text()
900 assert "line1\n" in s
900 assert "line1\n" in s
901 assert "line2" in s
901 assert "line2" in s
902
902
903
903
904 def test_file_var_expand():
904 def test_file_var_expand():
905 """%%writefile $filename"""
905 """%%writefile $filename"""
906 ip = get_ipython()
906 ip = get_ipython()
907 with TemporaryDirectory() as td:
907 with TemporaryDirectory() as td:
908 fname = os.path.join(td, 'file1')
908 fname = os.path.join(td, 'file1')
909 ip.user_ns['filename'] = fname
909 ip.user_ns['filename'] = fname
910 ip.run_cell_magic("writefile", '$filename', u'\n'.join([
910 ip.run_cell_magic("writefile", '$filename', u'\n'.join([
911 'line1',
911 'line1',
912 'line2',
912 'line2',
913 ]))
913 ]))
914 s = Path(fname).read_text()
914 s = Path(fname).read_text()
915 assert "line1\n" in s
915 assert "line1\n" in s
916 assert "line2" in s
916 assert "line2" in s
917
917
918
918
919 def test_file_unicode():
919 def test_file_unicode():
920 """%%writefile with unicode cell"""
920 """%%writefile with unicode cell"""
921 ip = get_ipython()
921 ip = get_ipython()
922 with TemporaryDirectory() as td:
922 with TemporaryDirectory() as td:
923 fname = os.path.join(td, 'file1')
923 fname = os.path.join(td, 'file1')
924 ip.run_cell_magic("writefile", fname, u'\n'.join([
924 ip.run_cell_magic("writefile", fname, u'\n'.join([
925 u'linΓ©1',
925 u'linΓ©1',
926 u'linΓ©2',
926 u'linΓ©2',
927 ]))
927 ]))
928 with io.open(fname, encoding='utf-8') as f:
928 with io.open(fname, encoding='utf-8') as f:
929 s = f.read()
929 s = f.read()
930 assert "linΓ©1\n" in s
930 assert "linΓ©1\n" in s
931 assert "linΓ©2" in s
931 assert "linΓ©2" in s
932
932
933
933
934 def test_file_amend():
934 def test_file_amend():
935 """%%writefile -a amends files"""
935 """%%writefile -a amends files"""
936 ip = get_ipython()
936 ip = get_ipython()
937 with TemporaryDirectory() as td:
937 with TemporaryDirectory() as td:
938 fname = os.path.join(td, 'file2')
938 fname = os.path.join(td, 'file2')
939 ip.run_cell_magic("writefile", fname, u'\n'.join([
939 ip.run_cell_magic("writefile", fname, u'\n'.join([
940 'line1',
940 'line1',
941 'line2',
941 'line2',
942 ]))
942 ]))
943 ip.run_cell_magic("writefile", "-a %s" % fname, u'\n'.join([
943 ip.run_cell_magic("writefile", "-a %s" % fname, u'\n'.join([
944 'line3',
944 'line3',
945 'line4',
945 'line4',
946 ]))
946 ]))
947 s = Path(fname).read_text()
947 s = Path(fname).read_text()
948 assert "line1\n" in s
948 assert "line1\n" in s
949 assert "line3\n" in s
949 assert "line3\n" in s
950
950
951
951
952 def test_file_spaces():
952 def test_file_spaces():
953 """%%file with spaces in filename"""
953 """%%file with spaces in filename"""
954 ip = get_ipython()
954 ip = get_ipython()
955 with TemporaryWorkingDirectory() as td:
955 with TemporaryWorkingDirectory() as td:
956 fname = "file name"
956 fname = "file name"
957 ip.run_cell_magic("file", '"%s"'%fname, u'\n'.join([
957 ip.run_cell_magic("file", '"%s"'%fname, u'\n'.join([
958 'line1',
958 'line1',
959 'line2',
959 'line2',
960 ]))
960 ]))
961 s = Path(fname).read_text()
961 s = Path(fname).read_text()
962 assert "line1\n" in s
962 assert "line1\n" in s
963 assert "line2" in s
963 assert "line2" in s
964
964
965
965
966 def test_script_config():
966 def test_script_config():
967 ip = get_ipython()
967 ip = get_ipython()
968 ip.config.ScriptMagics.script_magics = ['whoda']
968 ip.config.ScriptMagics.script_magics = ['whoda']
969 sm = script.ScriptMagics(shell=ip)
969 sm = script.ScriptMagics(shell=ip)
970 assert "whoda" in sm.magics["cell"]
970 assert "whoda" in sm.magics["cell"]
971
971
972
972
973 @pytest.fixture
973 def test_script_out():
974 def event_loop():
975 policy = asyncio.get_event_loop_policy()
976 loop = policy.new_event_loop()
977 policy.set_event_loop(loop)
978 yield loop
979 loop.close()
980
981
982 @dec.skip_win32
983 @pytest.mark.skipif(
984 sys.platform == "win32", reason="This test does not run under Windows"
985 )
986 def test_script_out(event_loop):
987 assert event_loop.is_running() is False
988
989 ip = get_ipython()
974 ip = get_ipython()
990 ip.run_cell_magic("script", "--out output sh", "echo 'hi'")
975 ip.run_cell_magic("script", f"--out output {sys.executable}", "print('hi')")
991 assert event_loop.is_running() is False
976 assert ip.user_ns["output"].strip() == "hi"
992 assert ip.user_ns["output"] == "hi\n"
993
977
994
978
995 @dec.skip_win32
979 def test_script_err():
996 @pytest.mark.skipif(
997 sys.platform == "win32", reason="This test does not run under Windows"
998 )
999 def test_script_err(event_loop):
1000 ip = get_ipython()
980 ip = get_ipython()
1001 assert event_loop.is_running() is False
981 ip.run_cell_magic(
1002 ip.run_cell_magic("script", "--err error sh", "echo 'hello' >&2")
982 "script",
1003 assert event_loop.is_running() is False
983 f"--err error {sys.executable}",
1004 assert ip.user_ns["error"] == "hello\n"
984 "import sys; print('hello', file=sys.stderr)",
985 )
986 assert ip.user_ns["error"].strip() == "hello"
1005
987
1006
988
1007 @dec.skip_win32
1008 @pytest.mark.skipif(
1009 sys.platform == "win32", reason="This test does not run under Windows"
1010 )
1011 def test_script_out_err():
989 def test_script_out_err():
1012
990
1013 ip = get_ipython()
991 ip = get_ipython()
1014 ip.run_cell_magic(
992 ip.run_cell_magic(
1015 "script", "--out output --err error sh", "echo 'hi'\necho 'hello' >&2"
993 "script",
994 f"--out output --err error {sys.executable}",
995 "\n".join(
996 [
997 "import sys",
998 "print('hi')",
999 "print('hello', file=sys.stderr)",
1000 ]
1001 ),
1016 )
1002 )
1017 assert ip.user_ns["output"] == "hi\n"
1003 assert ip.user_ns["output"].strip() == "hi"
1018 assert ip.user_ns["error"] == "hello\n"
1004 assert ip.user_ns["error"].strip() == "hello"
1019
1005
1020
1006
1021 @dec.skip_win32
1007 async def test_script_bg_out():
1022 @pytest.mark.skipif(
1023 sys.platform == "win32", reason="This test does not run under Windows"
1024 )
1025 async def test_script_bg_out(event_loop):
1026 ip = get_ipython()
1008 ip = get_ipython()
1027 ip.run_cell_magic("script", "--bg --out output sh", "echo 'hi'")
1009 ip.run_cell_magic("script", f"--bg --out output {sys.executable}", "print('hi')")
1028 assert (await ip.user_ns["output"].read()) == b"hi\n"
1010 assert (await ip.user_ns["output"].read()).strip() == b"hi"
1029 ip.user_ns["output"].close()
1011 assert ip.user_ns["output"].at_eof()
1030 event_loop.stop()
1031
1012
1032
1013
1033 @dec.skip_win32
1034 @pytest.mark.skipif(
1035 sys.platform == "win32", reason="This test does not run under Windows"
1036 )
1037 async def test_script_bg_err():
1014 async def test_script_bg_err():
1038 ip = get_ipython()
1015 ip = get_ipython()
1039 ip.run_cell_magic("script", "--bg --err error sh", "echo 'hello' >&2")
1016 ip.run_cell_magic(
1040 assert (await ip.user_ns["error"].read()) == b"hello\n"
1017 "script",
1041 ip.user_ns["error"].close()
1018 f"--bg --err error {sys.executable}",
1019 "import sys; print('hello', file=sys.stderr)",
1020 )
1021 assert (await ip.user_ns["error"].read()).strip() == b"hello"
1022 assert ip.user_ns["error"].at_eof()
1042
1023
1043
1024
1044 @dec.skip_win32
1045 @pytest.mark.skipif(
1046 sys.platform == "win32", reason="This test does not run under Windows"
1047 )
1048 async def test_script_bg_out_err():
1025 async def test_script_bg_out_err():
1049 ip = get_ipython()
1026 ip = get_ipython()
1050 ip.run_cell_magic(
1027 ip.run_cell_magic(
1051 "script", "--bg --out output --err error sh", "echo 'hi'\necho 'hello' >&2"
1028 "script",
1029 f"--bg --out output --err error {sys.executable}",
1030 "\n".join(
1031 [
1032 "import sys",
1033 "print('hi')",
1034 "print('hello', file=sys.stderr)",
1035 ]
1036 ),
1037 )
1038 assert (await ip.user_ns["output"].read()).strip() == b"hi"
1039 assert (await ip.user_ns["error"].read()).strip() == b"hello"
1040 assert ip.user_ns["output"].at_eof()
1041 assert ip.user_ns["error"].at_eof()
1042
1043
1044 async def test_script_bg_proc():
1045 ip = get_ipython()
1046 ip.run_cell_magic(
1047 "script",
1048 f"--bg --out output --proc p {sys.executable}",
1049 "\n".join(
1050 [
1051 "import sys",
1052 "print('hi')",
1053 "print('hello', file=sys.stderr)",
1054 ]
1055 ),
1052 )
1056 )
1053 assert (await ip.user_ns["output"].read()) == b"hi\n"
1057 p = ip.user_ns["p"]
1054 assert (await ip.user_ns["error"].read()) == b"hello\n"
1058 await p.wait()
1055 ip.user_ns["output"].close()
1059 assert p.returncode == 0
1056 ip.user_ns["error"].close()
1060 assert (await p.stdout.read()).strip() == b"hi"
1061 # not captured, so empty
1062 assert (await p.stderr.read()) == b""
1063 assert p.stdout.at_eof()
1064 assert p.stderr.at_eof()
1057
1065
1058
1066
1059 def test_script_defaults():
1067 def test_script_defaults():
1060 ip = get_ipython()
1068 ip = get_ipython()
1061 for cmd in ['sh', 'bash', 'perl', 'ruby']:
1069 for cmd in ['sh', 'bash', 'perl', 'ruby']:
1062 try:
1070 try:
1063 find_cmd(cmd)
1071 find_cmd(cmd)
1064 except Exception:
1072 except Exception:
1065 pass
1073 pass
1066 else:
1074 else:
1067 assert cmd in ip.magics_manager.magics["cell"]
1075 assert cmd in ip.magics_manager.magics["cell"]
1068
1076
1069
1077
1070 @magics_class
1078 @magics_class
1071 class FooFoo(Magics):
1079 class FooFoo(Magics):
1072 """class with both %foo and %%foo magics"""
1080 """class with both %foo and %%foo magics"""
1073 @line_magic('foo')
1081 @line_magic('foo')
1074 def line_foo(self, line):
1082 def line_foo(self, line):
1075 "I am line foo"
1083 "I am line foo"
1076 pass
1084 pass
1077
1085
1078 @cell_magic("foo")
1086 @cell_magic("foo")
1079 def cell_foo(self, line, cell):
1087 def cell_foo(self, line, cell):
1080 "I am cell foo, not line foo"
1088 "I am cell foo, not line foo"
1081 pass
1089 pass
1082
1090
1083 def test_line_cell_info():
1091 def test_line_cell_info():
1084 """%%foo and %foo magics are distinguishable to inspect"""
1092 """%%foo and %foo magics are distinguishable to inspect"""
1085 ip = get_ipython()
1093 ip = get_ipython()
1086 ip.magics_manager.register(FooFoo)
1094 ip.magics_manager.register(FooFoo)
1087 oinfo = ip.object_inspect("foo")
1095 oinfo = ip.object_inspect("foo")
1088 assert oinfo["found"] is True
1096 assert oinfo["found"] is True
1089 assert oinfo["ismagic"] is True
1097 assert oinfo["ismagic"] is True
1090
1098
1091 oinfo = ip.object_inspect("%%foo")
1099 oinfo = ip.object_inspect("%%foo")
1092 assert oinfo["found"] is True
1100 assert oinfo["found"] is True
1093 assert oinfo["ismagic"] is True
1101 assert oinfo["ismagic"] is True
1094 assert oinfo["docstring"] == FooFoo.cell_foo.__doc__
1102 assert oinfo["docstring"] == FooFoo.cell_foo.__doc__
1095
1103
1096 oinfo = ip.object_inspect("%foo")
1104 oinfo = ip.object_inspect("%foo")
1097 assert oinfo["found"] is True
1105 assert oinfo["found"] is True
1098 assert oinfo["ismagic"] is True
1106 assert oinfo["ismagic"] is True
1099 assert oinfo["docstring"] == FooFoo.line_foo.__doc__
1107 assert oinfo["docstring"] == FooFoo.line_foo.__doc__
1100
1108
1101
1109
1102 def test_multiple_magics():
1110 def test_multiple_magics():
1103 ip = get_ipython()
1111 ip = get_ipython()
1104 foo1 = FooFoo(ip)
1112 foo1 = FooFoo(ip)
1105 foo2 = FooFoo(ip)
1113 foo2 = FooFoo(ip)
1106 mm = ip.magics_manager
1114 mm = ip.magics_manager
1107 mm.register(foo1)
1115 mm.register(foo1)
1108 assert mm.magics["line"]["foo"].__self__ is foo1
1116 assert mm.magics["line"]["foo"].__self__ is foo1
1109 mm.register(foo2)
1117 mm.register(foo2)
1110 assert mm.magics["line"]["foo"].__self__ is foo2
1118 assert mm.magics["line"]["foo"].__self__ is foo2
1111
1119
1112
1120
1113 def test_alias_magic():
1121 def test_alias_magic():
1114 """Test %alias_magic."""
1122 """Test %alias_magic."""
1115 ip = get_ipython()
1123 ip = get_ipython()
1116 mm = ip.magics_manager
1124 mm = ip.magics_manager
1117
1125
1118 # Basic operation: both cell and line magics are created, if possible.
1126 # Basic operation: both cell and line magics are created, if possible.
1119 ip.run_line_magic("alias_magic", "timeit_alias timeit")
1127 ip.run_line_magic("alias_magic", "timeit_alias timeit")
1120 assert "timeit_alias" in mm.magics["line"]
1128 assert "timeit_alias" in mm.magics["line"]
1121 assert "timeit_alias" in mm.magics["cell"]
1129 assert "timeit_alias" in mm.magics["cell"]
1122
1130
1123 # --cell is specified, line magic not created.
1131 # --cell is specified, line magic not created.
1124 ip.run_line_magic("alias_magic", "--cell timeit_cell_alias timeit")
1132 ip.run_line_magic("alias_magic", "--cell timeit_cell_alias timeit")
1125 assert "timeit_cell_alias" not in mm.magics["line"]
1133 assert "timeit_cell_alias" not in mm.magics["line"]
1126 assert "timeit_cell_alias" in mm.magics["cell"]
1134 assert "timeit_cell_alias" in mm.magics["cell"]
1127
1135
1128 # Test that line alias is created successfully.
1136 # Test that line alias is created successfully.
1129 ip.run_line_magic("alias_magic", "--line env_alias env")
1137 ip.run_line_magic("alias_magic", "--line env_alias env")
1130 assert ip.run_line_magic("env", "") == ip.run_line_magic("env_alias", "")
1138 assert ip.run_line_magic("env", "") == ip.run_line_magic("env_alias", "")
1131
1139
1132 # Test that line alias with parameters passed in is created successfully.
1140 # Test that line alias with parameters passed in is created successfully.
1133 ip.run_line_magic(
1141 ip.run_line_magic(
1134 "alias_magic", "--line history_alias history --params " + shlex.quote("3")
1142 "alias_magic", "--line history_alias history --params " + shlex.quote("3")
1135 )
1143 )
1136 assert "history_alias" in mm.magics["line"]
1144 assert "history_alias" in mm.magics["line"]
1137
1145
1138
1146
1139 def test_save():
1147 def test_save():
1140 """Test %save."""
1148 """Test %save."""
1141 ip = get_ipython()
1149 ip = get_ipython()
1142 ip.history_manager.reset() # Clear any existing history.
1150 ip.history_manager.reset() # Clear any existing history.
1143 cmds = ["a=1", "def b():\n return a**2", "print(a, b())"]
1151 cmds = ["a=1", "def b():\n return a**2", "print(a, b())"]
1144 for i, cmd in enumerate(cmds, start=1):
1152 for i, cmd in enumerate(cmds, start=1):
1145 ip.history_manager.store_inputs(i, cmd)
1153 ip.history_manager.store_inputs(i, cmd)
1146 with TemporaryDirectory() as tmpdir:
1154 with TemporaryDirectory() as tmpdir:
1147 file = os.path.join(tmpdir, "testsave.py")
1155 file = os.path.join(tmpdir, "testsave.py")
1148 ip.run_line_magic("save", "%s 1-10" % file)
1156 ip.run_line_magic("save", "%s 1-10" % file)
1149 content = Path(file).read_text()
1157 content = Path(file).read_text()
1150 assert content.count(cmds[0]) == 1
1158 assert content.count(cmds[0]) == 1
1151 assert "coding: utf-8" in content
1159 assert "coding: utf-8" in content
1152 ip.run_line_magic("save", "-a %s 1-10" % file)
1160 ip.run_line_magic("save", "-a %s 1-10" % file)
1153 content = Path(file).read_text()
1161 content = Path(file).read_text()
1154 assert content.count(cmds[0]) == 2
1162 assert content.count(cmds[0]) == 2
1155 assert "coding: utf-8" in content
1163 assert "coding: utf-8" in content
1156
1164
1157
1165
1158 def test_save_with_no_args():
1166 def test_save_with_no_args():
1159 ip = get_ipython()
1167 ip = get_ipython()
1160 ip.history_manager.reset() # Clear any existing history.
1168 ip.history_manager.reset() # Clear any existing history.
1161 cmds = ["a=1", "def b():\n return a**2", "print(a, b())", "%save"]
1169 cmds = ["a=1", "def b():\n return a**2", "print(a, b())", "%save"]
1162 for i, cmd in enumerate(cmds, start=1):
1170 for i, cmd in enumerate(cmds, start=1):
1163 ip.history_manager.store_inputs(i, cmd)
1171 ip.history_manager.store_inputs(i, cmd)
1164
1172
1165 with TemporaryDirectory() as tmpdir:
1173 with TemporaryDirectory() as tmpdir:
1166 path = os.path.join(tmpdir, "testsave.py")
1174 path = os.path.join(tmpdir, "testsave.py")
1167 ip.run_line_magic("save", path)
1175 ip.run_line_magic("save", path)
1168 content = Path(path).read_text()
1176 content = Path(path).read_text()
1169 expected_content = dedent(
1177 expected_content = dedent(
1170 """\
1178 """\
1171 # coding: utf-8
1179 # coding: utf-8
1172 a=1
1180 a=1
1173 def b():
1181 def b():
1174 return a**2
1182 return a**2
1175 print(a, b())
1183 print(a, b())
1176 """
1184 """
1177 )
1185 )
1178 assert content == expected_content
1186 assert content == expected_content
1179
1187
1180
1188
1181 def test_store():
1189 def test_store():
1182 """Test %store."""
1190 """Test %store."""
1183 ip = get_ipython()
1191 ip = get_ipython()
1184 ip.run_line_magic('load_ext', 'storemagic')
1192 ip.run_line_magic('load_ext', 'storemagic')
1185
1193
1186 # make sure the storage is empty
1194 # make sure the storage is empty
1187 ip.run_line_magic("store", "-z")
1195 ip.run_line_magic("store", "-z")
1188 ip.user_ns["var"] = 42
1196 ip.user_ns["var"] = 42
1189 ip.run_line_magic("store", "var")
1197 ip.run_line_magic("store", "var")
1190 ip.user_ns["var"] = 39
1198 ip.user_ns["var"] = 39
1191 ip.run_line_magic("store", "-r")
1199 ip.run_line_magic("store", "-r")
1192 assert ip.user_ns["var"] == 42
1200 assert ip.user_ns["var"] == 42
1193
1201
1194 ip.run_line_magic("store", "-d var")
1202 ip.run_line_magic("store", "-d var")
1195 ip.user_ns["var"] = 39
1203 ip.user_ns["var"] = 39
1196 ip.run_line_magic("store", "-r")
1204 ip.run_line_magic("store", "-r")
1197 assert ip.user_ns["var"] == 39
1205 assert ip.user_ns["var"] == 39
1198
1206
1199
1207
1200 def _run_edit_test(arg_s, exp_filename=None,
1208 def _run_edit_test(arg_s, exp_filename=None,
1201 exp_lineno=-1,
1209 exp_lineno=-1,
1202 exp_contents=None,
1210 exp_contents=None,
1203 exp_is_temp=None):
1211 exp_is_temp=None):
1204 ip = get_ipython()
1212 ip = get_ipython()
1205 M = code.CodeMagics(ip)
1213 M = code.CodeMagics(ip)
1206 last_call = ['','']
1214 last_call = ['','']
1207 opts,args = M.parse_options(arg_s,'prxn:')
1215 opts,args = M.parse_options(arg_s,'prxn:')
1208 filename, lineno, is_temp = M._find_edit_target(ip, args, opts, last_call)
1216 filename, lineno, is_temp = M._find_edit_target(ip, args, opts, last_call)
1209
1217
1210 if exp_filename is not None:
1218 if exp_filename is not None:
1211 assert exp_filename == filename
1219 assert exp_filename == filename
1212 if exp_contents is not None:
1220 if exp_contents is not None:
1213 with io.open(filename, 'r', encoding='utf-8') as f:
1221 with io.open(filename, 'r', encoding='utf-8') as f:
1214 contents = f.read()
1222 contents = f.read()
1215 assert exp_contents == contents
1223 assert exp_contents == contents
1216 if exp_lineno != -1:
1224 if exp_lineno != -1:
1217 assert exp_lineno == lineno
1225 assert exp_lineno == lineno
1218 if exp_is_temp is not None:
1226 if exp_is_temp is not None:
1219 assert exp_is_temp == is_temp
1227 assert exp_is_temp == is_temp
1220
1228
1221
1229
1222 def test_edit_interactive():
1230 def test_edit_interactive():
1223 """%edit on interactively defined objects"""
1231 """%edit on interactively defined objects"""
1224 ip = get_ipython()
1232 ip = get_ipython()
1225 n = ip.execution_count
1233 n = ip.execution_count
1226 ip.run_cell("def foo(): return 1", store_history=True)
1234 ip.run_cell("def foo(): return 1", store_history=True)
1227
1235
1228 with pytest.raises(code.InteractivelyDefined) as e:
1236 with pytest.raises(code.InteractivelyDefined) as e:
1229 _run_edit_test("foo")
1237 _run_edit_test("foo")
1230 assert e.value.index == n
1238 assert e.value.index == n
1231
1239
1232
1240
1233 def test_edit_cell():
1241 def test_edit_cell():
1234 """%edit [cell id]"""
1242 """%edit [cell id]"""
1235 ip = get_ipython()
1243 ip = get_ipython()
1236
1244
1237 ip.run_cell("def foo(): return 1", store_history=True)
1245 ip.run_cell("def foo(): return 1", store_history=True)
1238
1246
1239 # test
1247 # test
1240 _run_edit_test("1", exp_contents=ip.user_ns['In'][1], exp_is_temp=True)
1248 _run_edit_test("1", exp_contents=ip.user_ns['In'][1], exp_is_temp=True)
1241
1249
1242 def test_edit_fname():
1250 def test_edit_fname():
1243 """%edit file"""
1251 """%edit file"""
1244 # test
1252 # test
1245 _run_edit_test("test file.py", exp_filename="test file.py")
1253 _run_edit_test("test file.py", exp_filename="test file.py")
1246
1254
1247 def test_bookmark():
1255 def test_bookmark():
1248 ip = get_ipython()
1256 ip = get_ipython()
1249 ip.run_line_magic('bookmark', 'bmname')
1257 ip.run_line_magic('bookmark', 'bmname')
1250 with tt.AssertPrints('bmname'):
1258 with tt.AssertPrints('bmname'):
1251 ip.run_line_magic('bookmark', '-l')
1259 ip.run_line_magic('bookmark', '-l')
1252 ip.run_line_magic('bookmark', '-d bmname')
1260 ip.run_line_magic('bookmark', '-d bmname')
1253
1261
1254 def test_ls_magic():
1262 def test_ls_magic():
1255 ip = get_ipython()
1263 ip = get_ipython()
1256 json_formatter = ip.display_formatter.formatters['application/json']
1264 json_formatter = ip.display_formatter.formatters['application/json']
1257 json_formatter.enabled = True
1265 json_formatter.enabled = True
1258 lsmagic = ip.magic('lsmagic')
1266 lsmagic = ip.magic('lsmagic')
1259 with warnings.catch_warnings(record=True) as w:
1267 with warnings.catch_warnings(record=True) as w:
1260 j = json_formatter(lsmagic)
1268 j = json_formatter(lsmagic)
1261 assert sorted(j) == ["cell", "line"]
1269 assert sorted(j) == ["cell", "line"]
1262 assert w == [] # no warnings
1270 assert w == [] # no warnings
1263
1271
1264
1272
1265 def test_strip_initial_indent():
1273 def test_strip_initial_indent():
1266 def sii(s):
1274 def sii(s):
1267 lines = s.splitlines()
1275 lines = s.splitlines()
1268 return '\n'.join(code.strip_initial_indent(lines))
1276 return '\n'.join(code.strip_initial_indent(lines))
1269
1277
1270 assert sii(" a = 1\nb = 2") == "a = 1\nb = 2"
1278 assert sii(" a = 1\nb = 2") == "a = 1\nb = 2"
1271 assert sii(" a\n b\nc") == "a\n b\nc"
1279 assert sii(" a\n b\nc") == "a\n b\nc"
1272 assert sii("a\n b") == "a\n b"
1280 assert sii("a\n b") == "a\n b"
1273
1281
1274 def test_logging_magic_quiet_from_arg():
1282 def test_logging_magic_quiet_from_arg():
1275 _ip.config.LoggingMagics.quiet = False
1283 _ip.config.LoggingMagics.quiet = False
1276 lm = logging.LoggingMagics(shell=_ip)
1284 lm = logging.LoggingMagics(shell=_ip)
1277 with TemporaryDirectory() as td:
1285 with TemporaryDirectory() as td:
1278 try:
1286 try:
1279 with tt.AssertNotPrints(re.compile("Activating.*")):
1287 with tt.AssertNotPrints(re.compile("Activating.*")):
1280 lm.logstart('-q {}'.format(
1288 lm.logstart('-q {}'.format(
1281 os.path.join(td, "quiet_from_arg.log")))
1289 os.path.join(td, "quiet_from_arg.log")))
1282 finally:
1290 finally:
1283 _ip.logger.logstop()
1291 _ip.logger.logstop()
1284
1292
1285 def test_logging_magic_quiet_from_config():
1293 def test_logging_magic_quiet_from_config():
1286 _ip.config.LoggingMagics.quiet = True
1294 _ip.config.LoggingMagics.quiet = True
1287 lm = logging.LoggingMagics(shell=_ip)
1295 lm = logging.LoggingMagics(shell=_ip)
1288 with TemporaryDirectory() as td:
1296 with TemporaryDirectory() as td:
1289 try:
1297 try:
1290 with tt.AssertNotPrints(re.compile("Activating.*")):
1298 with tt.AssertNotPrints(re.compile("Activating.*")):
1291 lm.logstart(os.path.join(td, "quiet_from_config.log"))
1299 lm.logstart(os.path.join(td, "quiet_from_config.log"))
1292 finally:
1300 finally:
1293 _ip.logger.logstop()
1301 _ip.logger.logstop()
1294
1302
1295
1303
1296 def test_logging_magic_not_quiet():
1304 def test_logging_magic_not_quiet():
1297 _ip.config.LoggingMagics.quiet = False
1305 _ip.config.LoggingMagics.quiet = False
1298 lm = logging.LoggingMagics(shell=_ip)
1306 lm = logging.LoggingMagics(shell=_ip)
1299 with TemporaryDirectory() as td:
1307 with TemporaryDirectory() as td:
1300 try:
1308 try:
1301 with tt.AssertPrints(re.compile("Activating.*")):
1309 with tt.AssertPrints(re.compile("Activating.*")):
1302 lm.logstart(os.path.join(td, "not_quiet.log"))
1310 lm.logstart(os.path.join(td, "not_quiet.log"))
1303 finally:
1311 finally:
1304 _ip.logger.logstop()
1312 _ip.logger.logstop()
1305
1313
1306
1314
1307 def test_time_no_var_expand():
1315 def test_time_no_var_expand():
1308 _ip.user_ns['a'] = 5
1316 _ip.user_ns['a'] = 5
1309 _ip.user_ns['b'] = []
1317 _ip.user_ns['b'] = []
1310 _ip.magic('time b.append("{a}")')
1318 _ip.magic('time b.append("{a}")')
1311 assert _ip.user_ns['b'] == ['{a}']
1319 assert _ip.user_ns['b'] == ['{a}']
1312
1320
1313
1321
1314 # this is slow, put at the end for local testing.
1322 # this is slow, put at the end for local testing.
1315 def test_timeit_arguments():
1323 def test_timeit_arguments():
1316 "Test valid timeit arguments, should not cause SyntaxError (GH #1269)"
1324 "Test valid timeit arguments, should not cause SyntaxError (GH #1269)"
1317 _ip.magic("timeit -n1 -r1 a=('#')")
1325 _ip.magic("timeit -n1 -r1 a=('#')")
1318
1326
1319
1327
1320 TEST_MODULE = """
1328 TEST_MODULE = """
1321 print('Loaded my_tmp')
1329 print('Loaded my_tmp')
1322 if __name__ == "__main__":
1330 if __name__ == "__main__":
1323 print('I just ran a script')
1331 print('I just ran a script')
1324 """
1332 """
1325
1333
1326
1334
1327 def test_run_module_from_import_hook():
1335 def test_run_module_from_import_hook():
1328 "Test that a module can be loaded via an import hook"
1336 "Test that a module can be loaded via an import hook"
1329 with TemporaryDirectory() as tmpdir:
1337 with TemporaryDirectory() as tmpdir:
1330 fullpath = os.path.join(tmpdir, 'my_tmp.py')
1338 fullpath = os.path.join(tmpdir, 'my_tmp.py')
1331 Path(fullpath).write_text(TEST_MODULE)
1339 Path(fullpath).write_text(TEST_MODULE)
1332
1340
1333 import importlib.abc
1341 import importlib.abc
1334 import importlib.util
1342 import importlib.util
1335
1343
1336 class MyTempImporter(importlib.abc.MetaPathFinder, importlib.abc.SourceLoader):
1344 class MyTempImporter(importlib.abc.MetaPathFinder, importlib.abc.SourceLoader):
1337 def find_spec(self, fullname, path, target=None):
1345 def find_spec(self, fullname, path, target=None):
1338 if fullname == "my_tmp":
1346 if fullname == "my_tmp":
1339 return importlib.util.spec_from_loader(fullname, self)
1347 return importlib.util.spec_from_loader(fullname, self)
1340
1348
1341 def get_filename(self, fullname):
1349 def get_filename(self, fullname):
1342 assert fullname == "my_tmp"
1350 assert fullname == "my_tmp"
1343 return fullpath
1351 return fullpath
1344
1352
1345 def get_data(self, path):
1353 def get_data(self, path):
1346 assert Path(path).samefile(fullpath)
1354 assert Path(path).samefile(fullpath)
1347 return Path(fullpath).read_text()
1355 return Path(fullpath).read_text()
1348
1356
1349 sys.meta_path.insert(0, MyTempImporter())
1357 sys.meta_path.insert(0, MyTempImporter())
1350
1358
1351 with capture_output() as captured:
1359 with capture_output() as captured:
1352 _ip.magic("run -m my_tmp")
1360 _ip.magic("run -m my_tmp")
1353 _ip.run_cell("import my_tmp")
1361 _ip.run_cell("import my_tmp")
1354
1362
1355 output = "Loaded my_tmp\nI just ran a script\nLoaded my_tmp\n"
1363 output = "Loaded my_tmp\nI just ran a script\nLoaded my_tmp\n"
1356 assert output == captured.stdout
1364 assert output == captured.stdout
1357
1365
1358 sys.meta_path.pop(0)
1366 sys.meta_path.pop(0)
@@ -1,699 +1,700 b''
1 """IPython terminal interface using prompt_toolkit"""
1 """IPython terminal interface using prompt_toolkit"""
2
2
3 import asyncio
3 import asyncio
4 import os
4 import os
5 import sys
5 import sys
6 import warnings
6 import warnings
7 from warnings import warn
7 from warnings import warn
8
8
9 from IPython.core.async_helpers import get_asyncio_loop
9 from IPython.core.interactiveshell import InteractiveShell, InteractiveShellABC
10 from IPython.core.interactiveshell import InteractiveShell, InteractiveShellABC
10 from IPython.utils import io
11 from IPython.utils import io
11 from IPython.utils.py3compat import input
12 from IPython.utils.py3compat import input
12 from IPython.utils.terminal import toggle_set_term_title, set_term_title, restore_term_title
13 from IPython.utils.terminal import toggle_set_term_title, set_term_title, restore_term_title
13 from IPython.utils.process import abbrev_cwd
14 from IPython.utils.process import abbrev_cwd
14 from traitlets import (
15 from traitlets import (
15 Bool,
16 Bool,
16 Unicode,
17 Unicode,
17 Dict,
18 Dict,
18 Integer,
19 Integer,
19 observe,
20 observe,
20 Instance,
21 Instance,
21 Type,
22 Type,
22 default,
23 default,
23 Enum,
24 Enum,
24 Union,
25 Union,
25 Any,
26 Any,
26 validate,
27 validate,
27 Float,
28 Float,
28 )
29 )
29
30
30 from prompt_toolkit.auto_suggest import AutoSuggestFromHistory
31 from prompt_toolkit.auto_suggest import AutoSuggestFromHistory
31 from prompt_toolkit.enums import DEFAULT_BUFFER, EditingMode
32 from prompt_toolkit.enums import DEFAULT_BUFFER, EditingMode
32 from prompt_toolkit.filters import (HasFocus, Condition, IsDone)
33 from prompt_toolkit.filters import (HasFocus, Condition, IsDone)
33 from prompt_toolkit.formatted_text import PygmentsTokens
34 from prompt_toolkit.formatted_text import PygmentsTokens
34 from prompt_toolkit.history import InMemoryHistory
35 from prompt_toolkit.history import InMemoryHistory
35 from prompt_toolkit.layout.processors import ConditionalProcessor, HighlightMatchingBracketProcessor
36 from prompt_toolkit.layout.processors import ConditionalProcessor, HighlightMatchingBracketProcessor
36 from prompt_toolkit.output import ColorDepth
37 from prompt_toolkit.output import ColorDepth
37 from prompt_toolkit.patch_stdout import patch_stdout
38 from prompt_toolkit.patch_stdout import patch_stdout
38 from prompt_toolkit.shortcuts import PromptSession, CompleteStyle, print_formatted_text
39 from prompt_toolkit.shortcuts import PromptSession, CompleteStyle, print_formatted_text
39 from prompt_toolkit.styles import DynamicStyle, merge_styles
40 from prompt_toolkit.styles import DynamicStyle, merge_styles
40 from prompt_toolkit.styles.pygments import style_from_pygments_cls, style_from_pygments_dict
41 from prompt_toolkit.styles.pygments import style_from_pygments_cls, style_from_pygments_dict
41 from prompt_toolkit import __version__ as ptk_version
42 from prompt_toolkit import __version__ as ptk_version
42
43
43 from pygments.styles import get_style_by_name
44 from pygments.styles import get_style_by_name
44 from pygments.style import Style
45 from pygments.style import Style
45 from pygments.token import Token
46 from pygments.token import Token
46
47
47 from .debugger import TerminalPdb, Pdb
48 from .debugger import TerminalPdb, Pdb
48 from .magics import TerminalMagics
49 from .magics import TerminalMagics
49 from .pt_inputhooks import get_inputhook_name_and_func
50 from .pt_inputhooks import get_inputhook_name_and_func
50 from .prompts import Prompts, ClassicPrompts, RichPromptDisplayHook
51 from .prompts import Prompts, ClassicPrompts, RichPromptDisplayHook
51 from .ptutils import IPythonPTCompleter, IPythonPTLexer
52 from .ptutils import IPythonPTCompleter, IPythonPTLexer
52 from .shortcuts import create_ipython_shortcuts
53 from .shortcuts import create_ipython_shortcuts
53
54
54 PTK3 = ptk_version.startswith('3.')
55 PTK3 = ptk_version.startswith('3.')
55
56
56
57
57 class _NoStyle(Style): pass
58 class _NoStyle(Style): pass
58
59
59
60
60
61
61 _style_overrides_light_bg = {
62 _style_overrides_light_bg = {
62 Token.Prompt: '#ansibrightblue',
63 Token.Prompt: '#ansibrightblue',
63 Token.PromptNum: '#ansiblue bold',
64 Token.PromptNum: '#ansiblue bold',
64 Token.OutPrompt: '#ansibrightred',
65 Token.OutPrompt: '#ansibrightred',
65 Token.OutPromptNum: '#ansired bold',
66 Token.OutPromptNum: '#ansired bold',
66 }
67 }
67
68
68 _style_overrides_linux = {
69 _style_overrides_linux = {
69 Token.Prompt: '#ansibrightgreen',
70 Token.Prompt: '#ansibrightgreen',
70 Token.PromptNum: '#ansigreen bold',
71 Token.PromptNum: '#ansigreen bold',
71 Token.OutPrompt: '#ansibrightred',
72 Token.OutPrompt: '#ansibrightred',
72 Token.OutPromptNum: '#ansired bold',
73 Token.OutPromptNum: '#ansired bold',
73 }
74 }
74
75
75 def get_default_editor():
76 def get_default_editor():
76 try:
77 try:
77 return os.environ['EDITOR']
78 return os.environ['EDITOR']
78 except KeyError:
79 except KeyError:
79 pass
80 pass
80 except UnicodeError:
81 except UnicodeError:
81 warn("$EDITOR environment variable is not pure ASCII. Using platform "
82 warn("$EDITOR environment variable is not pure ASCII. Using platform "
82 "default editor.")
83 "default editor.")
83
84
84 if os.name == 'posix':
85 if os.name == 'posix':
85 return 'vi' # the only one guaranteed to be there!
86 return 'vi' # the only one guaranteed to be there!
86 else:
87 else:
87 return 'notepad' # same in Windows!
88 return 'notepad' # same in Windows!
88
89
89 # conservatively check for tty
90 # conservatively check for tty
90 # overridden streams can result in things like:
91 # overridden streams can result in things like:
91 # - sys.stdin = None
92 # - sys.stdin = None
92 # - no isatty method
93 # - no isatty method
93 for _name in ('stdin', 'stdout', 'stderr'):
94 for _name in ('stdin', 'stdout', 'stderr'):
94 _stream = getattr(sys, _name)
95 _stream = getattr(sys, _name)
95 if not _stream or not hasattr(_stream, 'isatty') or not _stream.isatty():
96 if not _stream or not hasattr(_stream, 'isatty') or not _stream.isatty():
96 _is_tty = False
97 _is_tty = False
97 break
98 break
98 else:
99 else:
99 _is_tty = True
100 _is_tty = True
100
101
101
102
102 _use_simple_prompt = ('IPY_TEST_SIMPLE_PROMPT' in os.environ) or (not _is_tty)
103 _use_simple_prompt = ('IPY_TEST_SIMPLE_PROMPT' in os.environ) or (not _is_tty)
103
104
104 def black_reformat_handler(text_before_cursor):
105 def black_reformat_handler(text_before_cursor):
105 """
106 """
106 We do not need to protect against error,
107 We do not need to protect against error,
107 this is taken care at a higher level where any reformat error is ignored.
108 this is taken care at a higher level where any reformat error is ignored.
108 Indeed we may call reformatting on incomplete code.
109 Indeed we may call reformatting on incomplete code.
109 """
110 """
110 import black
111 import black
111
112
112 formatted_text = black.format_str(text_before_cursor, mode=black.FileMode())
113 formatted_text = black.format_str(text_before_cursor, mode=black.FileMode())
113 if not text_before_cursor.endswith("\n") and formatted_text.endswith("\n"):
114 if not text_before_cursor.endswith("\n") and formatted_text.endswith("\n"):
114 formatted_text = formatted_text[:-1]
115 formatted_text = formatted_text[:-1]
115 return formatted_text
116 return formatted_text
116
117
117
118
118 class TerminalInteractiveShell(InteractiveShell):
119 class TerminalInteractiveShell(InteractiveShell):
119 mime_renderers = Dict().tag(config=True)
120 mime_renderers = Dict().tag(config=True)
120
121
121 space_for_menu = Integer(6, help='Number of line at the bottom of the screen '
122 space_for_menu = Integer(6, help='Number of line at the bottom of the screen '
122 'to reserve for the tab completion menu, '
123 'to reserve for the tab completion menu, '
123 'search history, ...etc, the height of '
124 'search history, ...etc, the height of '
124 'these menus will at most this value. '
125 'these menus will at most this value. '
125 'Increase it is you prefer long and skinny '
126 'Increase it is you prefer long and skinny '
126 'menus, decrease for short and wide.'
127 'menus, decrease for short and wide.'
127 ).tag(config=True)
128 ).tag(config=True)
128
129
129 pt_app = None
130 pt_app = None
130 debugger_history = None
131 debugger_history = None
131
132
132 debugger_history_file = Unicode(
133 debugger_history_file = Unicode(
133 "~/.pdbhistory", help="File in which to store and read history"
134 "~/.pdbhistory", help="File in which to store and read history"
134 ).tag(config=True)
135 ).tag(config=True)
135
136
136 simple_prompt = Bool(_use_simple_prompt,
137 simple_prompt = Bool(_use_simple_prompt,
137 help="""Use `raw_input` for the REPL, without completion and prompt colors.
138 help="""Use `raw_input` for the REPL, without completion and prompt colors.
138
139
139 Useful when controlling IPython as a subprocess, and piping STDIN/OUT/ERR. Known usage are:
140 Useful when controlling IPython as a subprocess, and piping STDIN/OUT/ERR. Known usage are:
140 IPython own testing machinery, and emacs inferior-shell integration through elpy.
141 IPython own testing machinery, and emacs inferior-shell integration through elpy.
141
142
142 This mode default to `True` if the `IPY_TEST_SIMPLE_PROMPT`
143 This mode default to `True` if the `IPY_TEST_SIMPLE_PROMPT`
143 environment variable is set, or the current terminal is not a tty."""
144 environment variable is set, or the current terminal is not a tty."""
144 ).tag(config=True)
145 ).tag(config=True)
145
146
146 @property
147 @property
147 def debugger_cls(self):
148 def debugger_cls(self):
148 return Pdb if self.simple_prompt else TerminalPdb
149 return Pdb if self.simple_prompt else TerminalPdb
149
150
150 confirm_exit = Bool(True,
151 confirm_exit = Bool(True,
151 help="""
152 help="""
152 Set to confirm when you try to exit IPython with an EOF (Control-D
153 Set to confirm when you try to exit IPython with an EOF (Control-D
153 in Unix, Control-Z/Enter in Windows). By typing 'exit' or 'quit',
154 in Unix, Control-Z/Enter in Windows). By typing 'exit' or 'quit',
154 you can force a direct exit without any confirmation.""",
155 you can force a direct exit without any confirmation.""",
155 ).tag(config=True)
156 ).tag(config=True)
156
157
157 editing_mode = Unicode('emacs',
158 editing_mode = Unicode('emacs',
158 help="Shortcut style to use at the prompt. 'vi' or 'emacs'.",
159 help="Shortcut style to use at the prompt. 'vi' or 'emacs'.",
159 ).tag(config=True)
160 ).tag(config=True)
160
161
161 emacs_bindings_in_vi_insert_mode = Bool(
162 emacs_bindings_in_vi_insert_mode = Bool(
162 True,
163 True,
163 help="Add shortcuts from 'emacs' insert mode to 'vi' insert mode.",
164 help="Add shortcuts from 'emacs' insert mode to 'vi' insert mode.",
164 ).tag(config=True)
165 ).tag(config=True)
165
166
166 modal_cursor = Bool(
167 modal_cursor = Bool(
167 True,
168 True,
168 help="""
169 help="""
169 Cursor shape changes depending on vi mode: beam in vi insert mode,
170 Cursor shape changes depending on vi mode: beam in vi insert mode,
170 block in nav mode, underscore in replace mode.""",
171 block in nav mode, underscore in replace mode.""",
171 ).tag(config=True)
172 ).tag(config=True)
172
173
173 ttimeoutlen = Float(
174 ttimeoutlen = Float(
174 0.01,
175 0.01,
175 help="""The time in milliseconds that is waited for a key code
176 help="""The time in milliseconds that is waited for a key code
176 to complete.""",
177 to complete.""",
177 ).tag(config=True)
178 ).tag(config=True)
178
179
179 timeoutlen = Float(
180 timeoutlen = Float(
180 0.5,
181 0.5,
181 help="""The time in milliseconds that is waited for a mapped key
182 help="""The time in milliseconds that is waited for a mapped key
182 sequence to complete.""",
183 sequence to complete.""",
183 ).tag(config=True)
184 ).tag(config=True)
184
185
185 autoformatter = Unicode(
186 autoformatter = Unicode(
186 "black",
187 "black",
187 help="Autoformatter to reformat Terminal code. Can be `'black'` or `None`",
188 help="Autoformatter to reformat Terminal code. Can be `'black'` or `None`",
188 allow_none=True
189 allow_none=True
189 ).tag(config=True)
190 ).tag(config=True)
190
191
191 auto_match = Bool(
192 auto_match = Bool(
192 False,
193 False,
193 help="""
194 help="""
194 Automatically add/delete closing bracket or quote when opening bracket or quote is entered/deleted.
195 Automatically add/delete closing bracket or quote when opening bracket or quote is entered/deleted.
195 Brackets: (), [], {}
196 Brackets: (), [], {}
196 Quotes: '', \"\"
197 Quotes: '', \"\"
197 """,
198 """,
198 ).tag(config=True)
199 ).tag(config=True)
199
200
200 mouse_support = Bool(False,
201 mouse_support = Bool(False,
201 help="Enable mouse support in the prompt\n(Note: prevents selecting text with the mouse)"
202 help="Enable mouse support in the prompt\n(Note: prevents selecting text with the mouse)"
202 ).tag(config=True)
203 ).tag(config=True)
203
204
204 # We don't load the list of styles for the help string, because loading
205 # We don't load the list of styles for the help string, because loading
205 # Pygments plugins takes time and can cause unexpected errors.
206 # Pygments plugins takes time and can cause unexpected errors.
206 highlighting_style = Union([Unicode('legacy'), Type(klass=Style)],
207 highlighting_style = Union([Unicode('legacy'), Type(klass=Style)],
207 help="""The name or class of a Pygments style to use for syntax
208 help="""The name or class of a Pygments style to use for syntax
208 highlighting. To see available styles, run `pygmentize -L styles`."""
209 highlighting. To see available styles, run `pygmentize -L styles`."""
209 ).tag(config=True)
210 ).tag(config=True)
210
211
211 @validate('editing_mode')
212 @validate('editing_mode')
212 def _validate_editing_mode(self, proposal):
213 def _validate_editing_mode(self, proposal):
213 if proposal['value'].lower() == 'vim':
214 if proposal['value'].lower() == 'vim':
214 proposal['value']= 'vi'
215 proposal['value']= 'vi'
215 elif proposal['value'].lower() == 'default':
216 elif proposal['value'].lower() == 'default':
216 proposal['value']= 'emacs'
217 proposal['value']= 'emacs'
217
218
218 if hasattr(EditingMode, proposal['value'].upper()):
219 if hasattr(EditingMode, proposal['value'].upper()):
219 return proposal['value'].lower()
220 return proposal['value'].lower()
220
221
221 return self.editing_mode
222 return self.editing_mode
222
223
223
224
224 @observe('editing_mode')
225 @observe('editing_mode')
225 def _editing_mode(self, change):
226 def _editing_mode(self, change):
226 if self.pt_app:
227 if self.pt_app:
227 self.pt_app.editing_mode = getattr(EditingMode, change.new.upper())
228 self.pt_app.editing_mode = getattr(EditingMode, change.new.upper())
228
229
229 def _set_formatter(self, formatter):
230 def _set_formatter(self, formatter):
230 if formatter is None:
231 if formatter is None:
231 self.reformat_handler = lambda x:x
232 self.reformat_handler = lambda x:x
232 elif formatter == 'black':
233 elif formatter == 'black':
233 self.reformat_handler = black_reformat_handler
234 self.reformat_handler = black_reformat_handler
234 else:
235 else:
235 raise ValueError
236 raise ValueError
236
237
237 @observe("autoformatter")
238 @observe("autoformatter")
238 def _autoformatter_changed(self, change):
239 def _autoformatter_changed(self, change):
239 formatter = change.new
240 formatter = change.new
240 self._set_formatter(formatter)
241 self._set_formatter(formatter)
241
242
242 @observe('highlighting_style')
243 @observe('highlighting_style')
243 @observe('colors')
244 @observe('colors')
244 def _highlighting_style_changed(self, change):
245 def _highlighting_style_changed(self, change):
245 self.refresh_style()
246 self.refresh_style()
246
247
247 def refresh_style(self):
248 def refresh_style(self):
248 self._style = self._make_style_from_name_or_cls(self.highlighting_style)
249 self._style = self._make_style_from_name_or_cls(self.highlighting_style)
249
250
250
251
251 highlighting_style_overrides = Dict(
252 highlighting_style_overrides = Dict(
252 help="Override highlighting format for specific tokens"
253 help="Override highlighting format for specific tokens"
253 ).tag(config=True)
254 ).tag(config=True)
254
255
255 true_color = Bool(False,
256 true_color = Bool(False,
256 help="""Use 24bit colors instead of 256 colors in prompt highlighting.
257 help="""Use 24bit colors instead of 256 colors in prompt highlighting.
257 If your terminal supports true color, the following command should
258 If your terminal supports true color, the following command should
258 print ``TRUECOLOR`` in orange::
259 print ``TRUECOLOR`` in orange::
259
260
260 printf \"\\x1b[38;2;255;100;0mTRUECOLOR\\x1b[0m\\n\"
261 printf \"\\x1b[38;2;255;100;0mTRUECOLOR\\x1b[0m\\n\"
261 """,
262 """,
262 ).tag(config=True)
263 ).tag(config=True)
263
264
264 editor = Unicode(get_default_editor(),
265 editor = Unicode(get_default_editor(),
265 help="Set the editor used by IPython (default to $EDITOR/vi/notepad)."
266 help="Set the editor used by IPython (default to $EDITOR/vi/notepad)."
266 ).tag(config=True)
267 ).tag(config=True)
267
268
268 prompts_class = Type(Prompts, help='Class used to generate Prompt token for prompt_toolkit').tag(config=True)
269 prompts_class = Type(Prompts, help='Class used to generate Prompt token for prompt_toolkit').tag(config=True)
269
270
270 prompts = Instance(Prompts)
271 prompts = Instance(Prompts)
271
272
272 @default('prompts')
273 @default('prompts')
273 def _prompts_default(self):
274 def _prompts_default(self):
274 return self.prompts_class(self)
275 return self.prompts_class(self)
275
276
276 # @observe('prompts')
277 # @observe('prompts')
277 # def _(self, change):
278 # def _(self, change):
278 # self._update_layout()
279 # self._update_layout()
279
280
280 @default('displayhook_class')
281 @default('displayhook_class')
281 def _displayhook_class_default(self):
282 def _displayhook_class_default(self):
282 return RichPromptDisplayHook
283 return RichPromptDisplayHook
283
284
284 term_title = Bool(True,
285 term_title = Bool(True,
285 help="Automatically set the terminal title"
286 help="Automatically set the terminal title"
286 ).tag(config=True)
287 ).tag(config=True)
287
288
288 term_title_format = Unicode("IPython: {cwd}",
289 term_title_format = Unicode("IPython: {cwd}",
289 help="Customize the terminal title format. This is a python format string. " +
290 help="Customize the terminal title format. This is a python format string. " +
290 "Available substitutions are: {cwd}."
291 "Available substitutions are: {cwd}."
291 ).tag(config=True)
292 ).tag(config=True)
292
293
293 display_completions = Enum(('column', 'multicolumn','readlinelike'),
294 display_completions = Enum(('column', 'multicolumn','readlinelike'),
294 help= ( "Options for displaying tab completions, 'column', 'multicolumn', and "
295 help= ( "Options for displaying tab completions, 'column', 'multicolumn', and "
295 "'readlinelike'. These options are for `prompt_toolkit`, see "
296 "'readlinelike'. These options are for `prompt_toolkit`, see "
296 "`prompt_toolkit` documentation for more information."
297 "`prompt_toolkit` documentation for more information."
297 ),
298 ),
298 default_value='multicolumn').tag(config=True)
299 default_value='multicolumn').tag(config=True)
299
300
300 highlight_matching_brackets = Bool(True,
301 highlight_matching_brackets = Bool(True,
301 help="Highlight matching brackets.",
302 help="Highlight matching brackets.",
302 ).tag(config=True)
303 ).tag(config=True)
303
304
304 extra_open_editor_shortcuts = Bool(False,
305 extra_open_editor_shortcuts = Bool(False,
305 help="Enable vi (v) or Emacs (C-X C-E) shortcuts to open an external editor. "
306 help="Enable vi (v) or Emacs (C-X C-E) shortcuts to open an external editor. "
306 "This is in addition to the F2 binding, which is always enabled."
307 "This is in addition to the F2 binding, which is always enabled."
307 ).tag(config=True)
308 ).tag(config=True)
308
309
309 handle_return = Any(None,
310 handle_return = Any(None,
310 help="Provide an alternative handler to be called when the user presses "
311 help="Provide an alternative handler to be called when the user presses "
311 "Return. This is an advanced option intended for debugging, which "
312 "Return. This is an advanced option intended for debugging, which "
312 "may be changed or removed in later releases."
313 "may be changed or removed in later releases."
313 ).tag(config=True)
314 ).tag(config=True)
314
315
315 enable_history_search = Bool(True,
316 enable_history_search = Bool(True,
316 help="Allows to enable/disable the prompt toolkit history search"
317 help="Allows to enable/disable the prompt toolkit history search"
317 ).tag(config=True)
318 ).tag(config=True)
318
319
319 prompt_includes_vi_mode = Bool(True,
320 prompt_includes_vi_mode = Bool(True,
320 help="Display the current vi mode (when using vi editing mode)."
321 help="Display the current vi mode (when using vi editing mode)."
321 ).tag(config=True)
322 ).tag(config=True)
322
323
323 @observe('term_title')
324 @observe('term_title')
324 def init_term_title(self, change=None):
325 def init_term_title(self, change=None):
325 # Enable or disable the terminal title.
326 # Enable or disable the terminal title.
326 if self.term_title:
327 if self.term_title:
327 toggle_set_term_title(True)
328 toggle_set_term_title(True)
328 set_term_title(self.term_title_format.format(cwd=abbrev_cwd()))
329 set_term_title(self.term_title_format.format(cwd=abbrev_cwd()))
329 else:
330 else:
330 toggle_set_term_title(False)
331 toggle_set_term_title(False)
331
332
332 def restore_term_title(self):
333 def restore_term_title(self):
333 if self.term_title:
334 if self.term_title:
334 restore_term_title()
335 restore_term_title()
335
336
336 def init_display_formatter(self):
337 def init_display_formatter(self):
337 super(TerminalInteractiveShell, self).init_display_formatter()
338 super(TerminalInteractiveShell, self).init_display_formatter()
338 # terminal only supports plain text
339 # terminal only supports plain text
339 self.display_formatter.active_types = ["text/plain"]
340 self.display_formatter.active_types = ["text/plain"]
340
341
341 def init_prompt_toolkit_cli(self):
342 def init_prompt_toolkit_cli(self):
342 if self.simple_prompt:
343 if self.simple_prompt:
343 # Fall back to plain non-interactive output for tests.
344 # Fall back to plain non-interactive output for tests.
344 # This is very limited.
345 # This is very limited.
345 def prompt():
346 def prompt():
346 prompt_text = "".join(x[1] for x in self.prompts.in_prompt_tokens())
347 prompt_text = "".join(x[1] for x in self.prompts.in_prompt_tokens())
347 lines = [input(prompt_text)]
348 lines = [input(prompt_text)]
348 prompt_continuation = "".join(x[1] for x in self.prompts.continuation_prompt_tokens())
349 prompt_continuation = "".join(x[1] for x in self.prompts.continuation_prompt_tokens())
349 while self.check_complete('\n'.join(lines))[0] == 'incomplete':
350 while self.check_complete('\n'.join(lines))[0] == 'incomplete':
350 lines.append( input(prompt_continuation) )
351 lines.append( input(prompt_continuation) )
351 return '\n'.join(lines)
352 return '\n'.join(lines)
352 self.prompt_for_code = prompt
353 self.prompt_for_code = prompt
353 return
354 return
354
355
355 # Set up keyboard shortcuts
356 # Set up keyboard shortcuts
356 key_bindings = create_ipython_shortcuts(self)
357 key_bindings = create_ipython_shortcuts(self)
357
358
358 # Pre-populate history from IPython's history database
359 # Pre-populate history from IPython's history database
359 history = InMemoryHistory()
360 history = InMemoryHistory()
360 last_cell = u""
361 last_cell = u""
361 for __, ___, cell in self.history_manager.get_tail(self.history_load_length,
362 for __, ___, cell in self.history_manager.get_tail(self.history_load_length,
362 include_latest=True):
363 include_latest=True):
363 # Ignore blank lines and consecutive duplicates
364 # Ignore blank lines and consecutive duplicates
364 cell = cell.rstrip()
365 cell = cell.rstrip()
365 if cell and (cell != last_cell):
366 if cell and (cell != last_cell):
366 history.append_string(cell)
367 history.append_string(cell)
367 last_cell = cell
368 last_cell = cell
368
369
369 self._style = self._make_style_from_name_or_cls(self.highlighting_style)
370 self._style = self._make_style_from_name_or_cls(self.highlighting_style)
370 self.style = DynamicStyle(lambda: self._style)
371 self.style = DynamicStyle(lambda: self._style)
371
372
372 editing_mode = getattr(EditingMode, self.editing_mode.upper())
373 editing_mode = getattr(EditingMode, self.editing_mode.upper())
373
374
374 self.pt_loop = asyncio.new_event_loop()
375 self.pt_loop = asyncio.new_event_loop()
375 self.pt_app = PromptSession(
376 self.pt_app = PromptSession(
376 auto_suggest=AutoSuggestFromHistory(),
377 auto_suggest=AutoSuggestFromHistory(),
377 editing_mode=editing_mode,
378 editing_mode=editing_mode,
378 key_bindings=key_bindings,
379 key_bindings=key_bindings,
379 history=history,
380 history=history,
380 completer=IPythonPTCompleter(shell=self),
381 completer=IPythonPTCompleter(shell=self),
381 enable_history_search=self.enable_history_search,
382 enable_history_search=self.enable_history_search,
382 style=self.style,
383 style=self.style,
383 include_default_pygments_style=False,
384 include_default_pygments_style=False,
384 mouse_support=self.mouse_support,
385 mouse_support=self.mouse_support,
385 enable_open_in_editor=self.extra_open_editor_shortcuts,
386 enable_open_in_editor=self.extra_open_editor_shortcuts,
386 color_depth=self.color_depth,
387 color_depth=self.color_depth,
387 tempfile_suffix=".py",
388 tempfile_suffix=".py",
388 **self._extra_prompt_options()
389 **self._extra_prompt_options()
389 )
390 )
390
391
391 def _make_style_from_name_or_cls(self, name_or_cls):
392 def _make_style_from_name_or_cls(self, name_or_cls):
392 """
393 """
393 Small wrapper that make an IPython compatible style from a style name
394 Small wrapper that make an IPython compatible style from a style name
394
395
395 We need that to add style for prompt ... etc.
396 We need that to add style for prompt ... etc.
396 """
397 """
397 style_overrides = {}
398 style_overrides = {}
398 if name_or_cls == 'legacy':
399 if name_or_cls == 'legacy':
399 legacy = self.colors.lower()
400 legacy = self.colors.lower()
400 if legacy == 'linux':
401 if legacy == 'linux':
401 style_cls = get_style_by_name('monokai')
402 style_cls = get_style_by_name('monokai')
402 style_overrides = _style_overrides_linux
403 style_overrides = _style_overrides_linux
403 elif legacy == 'lightbg':
404 elif legacy == 'lightbg':
404 style_overrides = _style_overrides_light_bg
405 style_overrides = _style_overrides_light_bg
405 style_cls = get_style_by_name('pastie')
406 style_cls = get_style_by_name('pastie')
406 elif legacy == 'neutral':
407 elif legacy == 'neutral':
407 # The default theme needs to be visible on both a dark background
408 # The default theme needs to be visible on both a dark background
408 # and a light background, because we can't tell what the terminal
409 # and a light background, because we can't tell what the terminal
409 # looks like. These tweaks to the default theme help with that.
410 # looks like. These tweaks to the default theme help with that.
410 style_cls = get_style_by_name('default')
411 style_cls = get_style_by_name('default')
411 style_overrides.update({
412 style_overrides.update({
412 Token.Number: '#ansigreen',
413 Token.Number: '#ansigreen',
413 Token.Operator: 'noinherit',
414 Token.Operator: 'noinherit',
414 Token.String: '#ansiyellow',
415 Token.String: '#ansiyellow',
415 Token.Name.Function: '#ansiblue',
416 Token.Name.Function: '#ansiblue',
416 Token.Name.Class: 'bold #ansiblue',
417 Token.Name.Class: 'bold #ansiblue',
417 Token.Name.Namespace: 'bold #ansiblue',
418 Token.Name.Namespace: 'bold #ansiblue',
418 Token.Name.Variable.Magic: '#ansiblue',
419 Token.Name.Variable.Magic: '#ansiblue',
419 Token.Prompt: '#ansigreen',
420 Token.Prompt: '#ansigreen',
420 Token.PromptNum: '#ansibrightgreen bold',
421 Token.PromptNum: '#ansibrightgreen bold',
421 Token.OutPrompt: '#ansired',
422 Token.OutPrompt: '#ansired',
422 Token.OutPromptNum: '#ansibrightred bold',
423 Token.OutPromptNum: '#ansibrightred bold',
423 })
424 })
424
425
425 # Hack: Due to limited color support on the Windows console
426 # Hack: Due to limited color support on the Windows console
426 # the prompt colors will be wrong without this
427 # the prompt colors will be wrong without this
427 if os.name == 'nt':
428 if os.name == 'nt':
428 style_overrides.update({
429 style_overrides.update({
429 Token.Prompt: '#ansidarkgreen',
430 Token.Prompt: '#ansidarkgreen',
430 Token.PromptNum: '#ansigreen bold',
431 Token.PromptNum: '#ansigreen bold',
431 Token.OutPrompt: '#ansidarkred',
432 Token.OutPrompt: '#ansidarkred',
432 Token.OutPromptNum: '#ansired bold',
433 Token.OutPromptNum: '#ansired bold',
433 })
434 })
434 elif legacy =='nocolor':
435 elif legacy =='nocolor':
435 style_cls=_NoStyle
436 style_cls=_NoStyle
436 style_overrides = {}
437 style_overrides = {}
437 else :
438 else :
438 raise ValueError('Got unknown colors: ', legacy)
439 raise ValueError('Got unknown colors: ', legacy)
439 else :
440 else :
440 if isinstance(name_or_cls, str):
441 if isinstance(name_or_cls, str):
441 style_cls = get_style_by_name(name_or_cls)
442 style_cls = get_style_by_name(name_or_cls)
442 else:
443 else:
443 style_cls = name_or_cls
444 style_cls = name_or_cls
444 style_overrides = {
445 style_overrides = {
445 Token.Prompt: '#ansigreen',
446 Token.Prompt: '#ansigreen',
446 Token.PromptNum: '#ansibrightgreen bold',
447 Token.PromptNum: '#ansibrightgreen bold',
447 Token.OutPrompt: '#ansired',
448 Token.OutPrompt: '#ansired',
448 Token.OutPromptNum: '#ansibrightred bold',
449 Token.OutPromptNum: '#ansibrightred bold',
449 }
450 }
450 style_overrides.update(self.highlighting_style_overrides)
451 style_overrides.update(self.highlighting_style_overrides)
451 style = merge_styles([
452 style = merge_styles([
452 style_from_pygments_cls(style_cls),
453 style_from_pygments_cls(style_cls),
453 style_from_pygments_dict(style_overrides),
454 style_from_pygments_dict(style_overrides),
454 ])
455 ])
455
456
456 return style
457 return style
457
458
458 @property
459 @property
459 def pt_complete_style(self):
460 def pt_complete_style(self):
460 return {
461 return {
461 'multicolumn': CompleteStyle.MULTI_COLUMN,
462 'multicolumn': CompleteStyle.MULTI_COLUMN,
462 'column': CompleteStyle.COLUMN,
463 'column': CompleteStyle.COLUMN,
463 'readlinelike': CompleteStyle.READLINE_LIKE,
464 'readlinelike': CompleteStyle.READLINE_LIKE,
464 }[self.display_completions]
465 }[self.display_completions]
465
466
466 @property
467 @property
467 def color_depth(self):
468 def color_depth(self):
468 return (ColorDepth.TRUE_COLOR if self.true_color else None)
469 return (ColorDepth.TRUE_COLOR if self.true_color else None)
469
470
470 def _extra_prompt_options(self):
471 def _extra_prompt_options(self):
471 """
472 """
472 Return the current layout option for the current Terminal InteractiveShell
473 Return the current layout option for the current Terminal InteractiveShell
473 """
474 """
474 def get_message():
475 def get_message():
475 return PygmentsTokens(self.prompts.in_prompt_tokens())
476 return PygmentsTokens(self.prompts.in_prompt_tokens())
476
477
477 if self.editing_mode == 'emacs':
478 if self.editing_mode == 'emacs':
478 # with emacs mode the prompt is (usually) static, so we call only
479 # with emacs mode the prompt is (usually) static, so we call only
479 # the function once. With VI mode it can toggle between [ins] and
480 # the function once. With VI mode it can toggle between [ins] and
480 # [nor] so we can't precompute.
481 # [nor] so we can't precompute.
481 # here I'm going to favor the default keybinding which almost
482 # here I'm going to favor the default keybinding which almost
482 # everybody uses to decrease CPU usage.
483 # everybody uses to decrease CPU usage.
483 # if we have issues with users with custom Prompts we can see how to
484 # if we have issues with users with custom Prompts we can see how to
484 # work around this.
485 # work around this.
485 get_message = get_message()
486 get_message = get_message()
486
487
487 options = {
488 options = {
488 'complete_in_thread': False,
489 'complete_in_thread': False,
489 'lexer':IPythonPTLexer(),
490 'lexer':IPythonPTLexer(),
490 'reserve_space_for_menu':self.space_for_menu,
491 'reserve_space_for_menu':self.space_for_menu,
491 'message': get_message,
492 'message': get_message,
492 'prompt_continuation': (
493 'prompt_continuation': (
493 lambda width, lineno, is_soft_wrap:
494 lambda width, lineno, is_soft_wrap:
494 PygmentsTokens(self.prompts.continuation_prompt_tokens(width))),
495 PygmentsTokens(self.prompts.continuation_prompt_tokens(width))),
495 'multiline': True,
496 'multiline': True,
496 'complete_style': self.pt_complete_style,
497 'complete_style': self.pt_complete_style,
497
498
498 # Highlight matching brackets, but only when this setting is
499 # Highlight matching brackets, but only when this setting is
499 # enabled, and only when the DEFAULT_BUFFER has the focus.
500 # enabled, and only when the DEFAULT_BUFFER has the focus.
500 'input_processors': [ConditionalProcessor(
501 'input_processors': [ConditionalProcessor(
501 processor=HighlightMatchingBracketProcessor(chars='[](){}'),
502 processor=HighlightMatchingBracketProcessor(chars='[](){}'),
502 filter=HasFocus(DEFAULT_BUFFER) & ~IsDone() &
503 filter=HasFocus(DEFAULT_BUFFER) & ~IsDone() &
503 Condition(lambda: self.highlight_matching_brackets))],
504 Condition(lambda: self.highlight_matching_brackets))],
504 }
505 }
505 if not PTK3:
506 if not PTK3:
506 options['inputhook'] = self.inputhook
507 options['inputhook'] = self.inputhook
507
508
508 return options
509 return options
509
510
510 def prompt_for_code(self):
511 def prompt_for_code(self):
511 if self.rl_next_input:
512 if self.rl_next_input:
512 default = self.rl_next_input
513 default = self.rl_next_input
513 self.rl_next_input = None
514 self.rl_next_input = None
514 else:
515 else:
515 default = ''
516 default = ''
516
517
517 # In order to make sure that asyncio code written in the
518 # In order to make sure that asyncio code written in the
518 # interactive shell doesn't interfere with the prompt, we run the
519 # interactive shell doesn't interfere with the prompt, we run the
519 # prompt in a different event loop.
520 # prompt in a different event loop.
520 # If we don't do this, people could spawn coroutine with a
521 # If we don't do this, people could spawn coroutine with a
521 # while/true inside which will freeze the prompt.
522 # while/true inside which will freeze the prompt.
522
523
523 policy = asyncio.get_event_loop_policy()
524 policy = asyncio.get_event_loop_policy()
524 try:
525 old_loop = get_asyncio_loop()
525 old_loop = policy.get_event_loop()
526
526 except RuntimeError:
527 # FIXME: prompt_toolkit is using the deprecated `asyncio.get_event_loop`
527 # This happens when the the event loop is closed,
528 # to get the current event loop.
528 # e.g. by calling `asyncio.run()`.
529 # This will probably be replaced by an attribute or input argument,
529 old_loop = None
530 # at which point we can stop calling the soon-to-be-deprecated `set_event_loop` here.
530
531 if old_loop is not self.pt_loop:
531 policy.set_event_loop(self.pt_loop)
532 policy.set_event_loop(self.pt_loop)
532 try:
533 try:
533 with patch_stdout(raw=True):
534 with patch_stdout(raw=True):
534 text = self.pt_app.prompt(
535 text = self.pt_app.prompt(
535 default=default,
536 default=default,
536 **self._extra_prompt_options())
537 **self._extra_prompt_options())
537 finally:
538 finally:
538 # Restore the original event loop.
539 # Restore the original event loop.
539 if old_loop is not None:
540 if old_loop is not None and old_loop is not self.pt_loop:
540 policy.set_event_loop(old_loop)
541 policy.set_event_loop(old_loop)
541
542
542 return text
543 return text
543
544
544 def enable_win_unicode_console(self):
545 def enable_win_unicode_console(self):
545 # Since IPython 7.10 doesn't support python < 3.6 and PEP 528, Python uses the unicode APIs for the Windows
546 # Since IPython 7.10 doesn't support python < 3.6 and PEP 528, Python uses the unicode APIs for the Windows
546 # console by default, so WUC shouldn't be needed.
547 # console by default, so WUC shouldn't be needed.
547 from warnings import warn
548 from warnings import warn
548 warn("`enable_win_unicode_console` is deprecated since IPython 7.10, does not do anything and will be removed in the future",
549 warn("`enable_win_unicode_console` is deprecated since IPython 7.10, does not do anything and will be removed in the future",
549 DeprecationWarning,
550 DeprecationWarning,
550 stacklevel=2)
551 stacklevel=2)
551
552
552 def init_io(self):
553 def init_io(self):
553 if sys.platform not in {'win32', 'cli'}:
554 if sys.platform not in {'win32', 'cli'}:
554 return
555 return
555
556
556 import colorama
557 import colorama
557 colorama.init()
558 colorama.init()
558
559
559 def init_magics(self):
560 def init_magics(self):
560 super(TerminalInteractiveShell, self).init_magics()
561 super(TerminalInteractiveShell, self).init_magics()
561 self.register_magics(TerminalMagics)
562 self.register_magics(TerminalMagics)
562
563
563 def init_alias(self):
564 def init_alias(self):
564 # The parent class defines aliases that can be safely used with any
565 # The parent class defines aliases that can be safely used with any
565 # frontend.
566 # frontend.
566 super(TerminalInteractiveShell, self).init_alias()
567 super(TerminalInteractiveShell, self).init_alias()
567
568
568 # Now define aliases that only make sense on the terminal, because they
569 # Now define aliases that only make sense on the terminal, because they
569 # need direct access to the console in a way that we can't emulate in
570 # need direct access to the console in a way that we can't emulate in
570 # GUI or web frontend
571 # GUI or web frontend
571 if os.name == 'posix':
572 if os.name == 'posix':
572 for cmd in ('clear', 'more', 'less', 'man'):
573 for cmd in ('clear', 'more', 'less', 'man'):
573 self.alias_manager.soft_define_alias(cmd, cmd)
574 self.alias_manager.soft_define_alias(cmd, cmd)
574
575
575
576
576 def __init__(self, *args, **kwargs):
577 def __init__(self, *args, **kwargs):
577 super(TerminalInteractiveShell, self).__init__(*args, **kwargs)
578 super(TerminalInteractiveShell, self).__init__(*args, **kwargs)
578 self.init_prompt_toolkit_cli()
579 self.init_prompt_toolkit_cli()
579 self.init_term_title()
580 self.init_term_title()
580 self.keep_running = True
581 self.keep_running = True
581 self._set_formatter(self.autoformatter)
582 self._set_formatter(self.autoformatter)
582
583
583
584
584 def ask_exit(self):
585 def ask_exit(self):
585 self.keep_running = False
586 self.keep_running = False
586
587
587 rl_next_input = None
588 rl_next_input = None
588
589
589 def interact(self):
590 def interact(self):
590 self.keep_running = True
591 self.keep_running = True
591 while self.keep_running:
592 while self.keep_running:
592 print(self.separate_in, end='')
593 print(self.separate_in, end='')
593
594
594 try:
595 try:
595 code = self.prompt_for_code()
596 code = self.prompt_for_code()
596 except EOFError:
597 except EOFError:
597 if (not self.confirm_exit) \
598 if (not self.confirm_exit) \
598 or self.ask_yes_no('Do you really want to exit ([y]/n)?','y','n'):
599 or self.ask_yes_no('Do you really want to exit ([y]/n)?','y','n'):
599 self.ask_exit()
600 self.ask_exit()
600
601
601 else:
602 else:
602 if code:
603 if code:
603 self.run_cell(code, store_history=True)
604 self.run_cell(code, store_history=True)
604
605
605 def mainloop(self):
606 def mainloop(self):
606 # An extra layer of protection in case someone mashing Ctrl-C breaks
607 # An extra layer of protection in case someone mashing Ctrl-C breaks
607 # out of our internal code.
608 # out of our internal code.
608 while True:
609 while True:
609 try:
610 try:
610 self.interact()
611 self.interact()
611 break
612 break
612 except KeyboardInterrupt as e:
613 except KeyboardInterrupt as e:
613 print("\n%s escaped interact()\n" % type(e).__name__)
614 print("\n%s escaped interact()\n" % type(e).__name__)
614 finally:
615 finally:
615 # An interrupt during the eventloop will mess up the
616 # An interrupt during the eventloop will mess up the
616 # internal state of the prompt_toolkit library.
617 # internal state of the prompt_toolkit library.
617 # Stopping the eventloop fixes this, see
618 # Stopping the eventloop fixes this, see
618 # https://github.com/ipython/ipython/pull/9867
619 # https://github.com/ipython/ipython/pull/9867
619 if hasattr(self, '_eventloop'):
620 if hasattr(self, '_eventloop'):
620 self._eventloop.stop()
621 self._eventloop.stop()
621
622
622 self.restore_term_title()
623 self.restore_term_title()
623
624
624 # try to call some at-exit operation optimistically as some things can't
625 # try to call some at-exit operation optimistically as some things can't
625 # be done during interpreter shutdown. this is technically inaccurate as
626 # be done during interpreter shutdown. this is technically inaccurate as
626 # this make mainlool not re-callable, but that should be a rare if not
627 # this make mainlool not re-callable, but that should be a rare if not
627 # in existent use case.
628 # in existent use case.
628
629
629 self._atexit_once()
630 self._atexit_once()
630
631
631
632
632 _inputhook = None
633 _inputhook = None
633 def inputhook(self, context):
634 def inputhook(self, context):
634 if self._inputhook is not None:
635 if self._inputhook is not None:
635 self._inputhook(context)
636 self._inputhook(context)
636
637
637 active_eventloop = None
638 active_eventloop = None
638 def enable_gui(self, gui=None):
639 def enable_gui(self, gui=None):
639 if gui and (gui != 'inline') :
640 if gui and (gui != 'inline') :
640 self.active_eventloop, self._inputhook =\
641 self.active_eventloop, self._inputhook =\
641 get_inputhook_name_and_func(gui)
642 get_inputhook_name_and_func(gui)
642 else:
643 else:
643 self.active_eventloop = self._inputhook = None
644 self.active_eventloop = self._inputhook = None
644
645
645 # For prompt_toolkit 3.0. We have to create an asyncio event loop with
646 # For prompt_toolkit 3.0. We have to create an asyncio event loop with
646 # this inputhook.
647 # this inputhook.
647 if PTK3:
648 if PTK3:
648 import asyncio
649 import asyncio
649 from prompt_toolkit.eventloop import new_eventloop_with_inputhook
650 from prompt_toolkit.eventloop import new_eventloop_with_inputhook
650
651
651 if gui == 'asyncio':
652 if gui == 'asyncio':
652 # When we integrate the asyncio event loop, run the UI in the
653 # When we integrate the asyncio event loop, run the UI in the
653 # same event loop as the rest of the code. don't use an actual
654 # same event loop as the rest of the code. don't use an actual
654 # input hook. (Asyncio is not made for nesting event loops.)
655 # input hook. (Asyncio is not made for nesting event loops.)
655 self.pt_loop = asyncio.get_event_loop_policy().get_event_loop()
656 self.pt_loop = get_asyncio_loop()
656
657
657 elif self._inputhook:
658 elif self._inputhook:
658 # If an inputhook was set, create a new asyncio event loop with
659 # If an inputhook was set, create a new asyncio event loop with
659 # this inputhook for the prompt.
660 # this inputhook for the prompt.
660 self.pt_loop = new_eventloop_with_inputhook(self._inputhook)
661 self.pt_loop = new_eventloop_with_inputhook(self._inputhook)
661 else:
662 else:
662 # When there's no inputhook, run the prompt in a separate
663 # When there's no inputhook, run the prompt in a separate
663 # asyncio event loop.
664 # asyncio event loop.
664 self.pt_loop = asyncio.new_event_loop()
665 self.pt_loop = asyncio.new_event_loop()
665
666
666 # Run !system commands directly, not through pipes, so terminal programs
667 # Run !system commands directly, not through pipes, so terminal programs
667 # work correctly.
668 # work correctly.
668 system = InteractiveShell.system_raw
669 system = InteractiveShell.system_raw
669
670
670 def auto_rewrite_input(self, cmd):
671 def auto_rewrite_input(self, cmd):
671 """Overridden from the parent class to use fancy rewriting prompt"""
672 """Overridden from the parent class to use fancy rewriting prompt"""
672 if not self.show_rewritten_input:
673 if not self.show_rewritten_input:
673 return
674 return
674
675
675 tokens = self.prompts.rewrite_prompt_tokens()
676 tokens = self.prompts.rewrite_prompt_tokens()
676 if self.pt_app:
677 if self.pt_app:
677 print_formatted_text(PygmentsTokens(tokens), end='',
678 print_formatted_text(PygmentsTokens(tokens), end='',
678 style=self.pt_app.app.style)
679 style=self.pt_app.app.style)
679 print(cmd)
680 print(cmd)
680 else:
681 else:
681 prompt = ''.join(s for t, s in tokens)
682 prompt = ''.join(s for t, s in tokens)
682 print(prompt, cmd, sep='')
683 print(prompt, cmd, sep='')
683
684
684 _prompts_before = None
685 _prompts_before = None
685 def switch_doctest_mode(self, mode):
686 def switch_doctest_mode(self, mode):
686 """Switch prompts to classic for %doctest_mode"""
687 """Switch prompts to classic for %doctest_mode"""
687 if mode:
688 if mode:
688 self._prompts_before = self.prompts
689 self._prompts_before = self.prompts
689 self.prompts = ClassicPrompts(self)
690 self.prompts = ClassicPrompts(self)
690 elif self._prompts_before:
691 elif self._prompts_before:
691 self.prompts = self._prompts_before
692 self.prompts = self._prompts_before
692 self._prompts_before = None
693 self._prompts_before = None
693 # self._update_layout()
694 # self._update_layout()
694
695
695
696
696 InteractiveShellABC.register(TerminalInteractiveShell)
697 InteractiveShellABC.register(TerminalInteractiveShell)
697
698
698 if __name__ == '__main__':
699 if __name__ == '__main__':
699 TerminalInteractiveShell.instance().interact()
700 TerminalInteractiveShell.instance().interact()
@@ -1,64 +1,63 b''
1 """
1 """
2 Inputhook for running the original asyncio event loop while we're waiting for
2 Inputhook for running the original asyncio event loop while we're waiting for
3 input.
3 input.
4
4
5 By default, in IPython, we run the prompt with a different asyncio event loop,
5 By default, in IPython, we run the prompt with a different asyncio event loop,
6 because otherwise we risk that people are freezing the prompt by scheduling bad
6 because otherwise we risk that people are freezing the prompt by scheduling bad
7 coroutines. E.g., a coroutine that does a while/true and never yield back
7 coroutines. E.g., a coroutine that does a while/true and never yield back
8 control to the loop. We can't cancel that.
8 control to the loop. We can't cancel that.
9
9
10 However, sometimes we want the asyncio loop to keep running while waiting for
10 However, sometimes we want the asyncio loop to keep running while waiting for
11 a prompt.
11 a prompt.
12
12
13 The following example will print the numbers from 1 to 10 above the prompt,
13 The following example will print the numbers from 1 to 10 above the prompt,
14 while we are waiting for input. (This works also because we use
14 while we are waiting for input. (This works also because we use
15 prompt_toolkit`s `patch_stdout`)::
15 prompt_toolkit`s `patch_stdout`)::
16
16
17 In [1]: import asyncio
17 In [1]: import asyncio
18
18
19 In [2]: %gui asyncio
19 In [2]: %gui asyncio
20
20
21 In [3]: async def f():
21 In [3]: async def f():
22 ...: for i in range(10):
22 ...: for i in range(10):
23 ...: await asyncio.sleep(1)
23 ...: await asyncio.sleep(1)
24 ...: print(i)
24 ...: print(i)
25
25
26
26
27 In [4]: asyncio.ensure_future(f())
27 In [4]: asyncio.ensure_future(f())
28
28
29 """
29 """
30 import asyncio
31 from prompt_toolkit import __version__ as ptk_version
30 from prompt_toolkit import __version__ as ptk_version
32
31
33 PTK3 = ptk_version.startswith('3.')
32 from IPython.core.async_helpers import get_asyncio_loop
34
33
34 PTK3 = ptk_version.startswith('3.')
35
35
36 # Keep reference to the original asyncio loop, because getting the event loop
37 # within the input hook would return the other loop.
38 loop = asyncio.get_event_loop_policy().get_event_loop()
39
36
40
37
41 def inputhook(context):
38 def inputhook(context):
42 """
39 """
43 Inputhook for asyncio event loop integration.
40 Inputhook for asyncio event loop integration.
44 """
41 """
45 # For prompt_toolkit 3.0, this input hook literally doesn't do anything.
42 # For prompt_toolkit 3.0, this input hook literally doesn't do anything.
46 # The event loop integration here is implemented in `interactiveshell.py`
43 # The event loop integration here is implemented in `interactiveshell.py`
47 # by running the prompt itself in the current asyncio loop. The main reason
44 # by running the prompt itself in the current asyncio loop. The main reason
48 # for this is that nesting asyncio event loops is unreliable.
45 # for this is that nesting asyncio event loops is unreliable.
49 if PTK3:
46 if PTK3:
50 return
47 return
51
48
52 # For prompt_toolkit 2.0, we can run the current asyncio event loop,
49 # For prompt_toolkit 2.0, we can run the current asyncio event loop,
53 # because prompt_toolkit 2.0 uses a different event loop internally.
50 # because prompt_toolkit 2.0 uses a different event loop internally.
54
51
52 # get the persistent asyncio event loop
53 loop = get_asyncio_loop()
54
55 def stop():
55 def stop():
56 loop.stop()
56 loop.stop()
57
57
58 fileno = context.fileno()
58 fileno = context.fileno()
59 loop.add_reader(fileno, stop)
59 loop.add_reader(fileno, stop)
60 try:
60 try:
61 loop.run_forever()
61 loop.run_forever()
62 finally:
62 finally:
63 loop.remove_reader(fileno)
63 loop.remove_reader(fileno)
64
@@ -1,184 +1,185 b''
1 # -*- coding: utf-8 -*-
1 # -*- coding: utf-8 -*-
2 """Setup script for IPython.
2 """Setup script for IPython.
3
3
4 Under Posix environments it works like a typical setup.py script.
4 Under Posix environments it works like a typical setup.py script.
5 Under Windows, the command sdist is not supported, since IPython
5 Under Windows, the command sdist is not supported, since IPython
6 requires utilities which are not available under Windows."""
6 requires utilities which are not available under Windows."""
7
7
8 #-----------------------------------------------------------------------------
8 #-----------------------------------------------------------------------------
9 # Copyright (c) 2008-2011, IPython Development Team.
9 # Copyright (c) 2008-2011, IPython Development Team.
10 # Copyright (c) 2001-2007, Fernando Perez <fernando.perez@colorado.edu>
10 # Copyright (c) 2001-2007, Fernando Perez <fernando.perez@colorado.edu>
11 # Copyright (c) 2001, Janko Hauser <jhauser@zscout.de>
11 # Copyright (c) 2001, Janko Hauser <jhauser@zscout.de>
12 # Copyright (c) 2001, Nathaniel Gray <n8gray@caltech.edu>
12 # Copyright (c) 2001, Nathaniel Gray <n8gray@caltech.edu>
13 #
13 #
14 # Distributed under the terms of the Modified BSD License.
14 # Distributed under the terms of the Modified BSD License.
15 #
15 #
16 # The full license is in the file COPYING.rst, distributed with this software.
16 # The full license is in the file COPYING.rst, distributed with this software.
17 #-----------------------------------------------------------------------------
17 #-----------------------------------------------------------------------------
18
18
19 import os
19 import os
20 import sys
20 import sys
21 from itertools import chain
21 from itertools import chain
22
22
23 # **Python version check**
23 # **Python version check**
24 #
24 #
25 # This check is also made in IPython/__init__, don't forget to update both when
25 # This check is also made in IPython/__init__, don't forget to update both when
26 # changing Python version requirements.
26 # changing Python version requirements.
27 if sys.version_info < (3, 8):
27 if sys.version_info < (3, 8):
28 pip_message = 'This may be due to an out of date pip. Make sure you have pip >= 9.0.1.'
28 pip_message = 'This may be due to an out of date pip. Make sure you have pip >= 9.0.1.'
29 try:
29 try:
30 import pip
30 import pip
31 pip_version = tuple([int(x) for x in pip.__version__.split('.')[:3]])
31 pip_version = tuple([int(x) for x in pip.__version__.split('.')[:3]])
32 if pip_version < (9, 0, 1) :
32 if pip_version < (9, 0, 1) :
33 pip_message = 'Your pip version is out of date, please install pip >= 9.0.1. '\
33 pip_message = 'Your pip version is out of date, please install pip >= 9.0.1. '\
34 'pip {} detected.'.format(pip.__version__)
34 'pip {} detected.'.format(pip.__version__)
35 else:
35 else:
36 # pip is new enough - it must be something else
36 # pip is new enough - it must be something else
37 pip_message = ''
37 pip_message = ''
38 except Exception:
38 except Exception:
39 pass
39 pass
40
40
41
41
42 error = """
42 error = """
43 IPython 8+ supports Python 3.8 and above, following NEP 29.
43 IPython 8+ supports Python 3.8 and above, following NEP 29.
44 When using Python 2.7, please install IPython 5.x LTS Long Term Support version.
44 When using Python 2.7, please install IPython 5.x LTS Long Term Support version.
45 Python 3.3 and 3.4 were supported up to IPython 6.x.
45 Python 3.3 and 3.4 were supported up to IPython 6.x.
46 Python 3.5 was supported with IPython 7.0 to 7.9.
46 Python 3.5 was supported with IPython 7.0 to 7.9.
47 Python 3.6 was supported with IPython up to 7.16.
47 Python 3.6 was supported with IPython up to 7.16.
48 Python 3.7 was still supported with the 7.x branch.
48 Python 3.7 was still supported with the 7.x branch.
49
49
50 See IPython `README.rst` file for more information:
50 See IPython `README.rst` file for more information:
51
51
52 https://github.com/ipython/ipython/blob/master/README.rst
52 https://github.com/ipython/ipython/blob/master/README.rst
53
53
54 Python {py} detected.
54 Python {py} detected.
55 {pip}
55 {pip}
56 """.format(py=sys.version_info, pip=pip_message )
56 """.format(py=sys.version_info, pip=pip_message )
57
57
58 print(error, file=sys.stderr)
58 print(error, file=sys.stderr)
59 sys.exit(1)
59 sys.exit(1)
60
60
61 # At least we're on the python version we need, move on.
61 # At least we're on the python version we need, move on.
62
62
63 from setuptools import setup
63 from setuptools import setup
64
64
65 # Our own imports
65 # Our own imports
66 from setupbase import target_update
66 from setupbase import target_update
67
67
68 from setupbase import (
68 from setupbase import (
69 setup_args,
69 setup_args,
70 check_package_data_first,
70 check_package_data_first,
71 find_data_files,
71 find_data_files,
72 git_prebuild,
72 git_prebuild,
73 install_symlinked,
73 install_symlinked,
74 install_lib_symlink,
74 install_lib_symlink,
75 install_scripts_for_symlink,
75 install_scripts_for_symlink,
76 unsymlink,
76 unsymlink,
77 )
77 )
78
78
79 #-------------------------------------------------------------------------------
79 #-------------------------------------------------------------------------------
80 # Handle OS specific things
80 # Handle OS specific things
81 #-------------------------------------------------------------------------------
81 #-------------------------------------------------------------------------------
82
82
83 if os.name in ('nt','dos'):
83 if os.name in ('nt','dos'):
84 os_name = 'windows'
84 os_name = 'windows'
85 else:
85 else:
86 os_name = os.name
86 os_name = os.name
87
87
88 # Under Windows, 'sdist' has not been supported. Now that the docs build with
88 # Under Windows, 'sdist' has not been supported. Now that the docs build with
89 # Sphinx it might work, but let's not turn it on until someone confirms that it
89 # Sphinx it might work, but let's not turn it on until someone confirms that it
90 # actually works.
90 # actually works.
91 if os_name == 'windows' and 'sdist' in sys.argv:
91 if os_name == 'windows' and 'sdist' in sys.argv:
92 print('The sdist command is not available under Windows. Exiting.')
92 print('The sdist command is not available under Windows. Exiting.')
93 sys.exit(1)
93 sys.exit(1)
94
94
95
95
96 #-------------------------------------------------------------------------------
96 #-------------------------------------------------------------------------------
97 # Things related to the IPython documentation
97 # Things related to the IPython documentation
98 #-------------------------------------------------------------------------------
98 #-------------------------------------------------------------------------------
99
99
100 # update the manuals when building a source dist
100 # update the manuals when building a source dist
101 if len(sys.argv) >= 2 and sys.argv[1] in ('sdist','bdist_rpm'):
101 if len(sys.argv) >= 2 and sys.argv[1] in ('sdist','bdist_rpm'):
102
102
103 # List of things to be updated. Each entry is a triplet of args for
103 # List of things to be updated. Each entry is a triplet of args for
104 # target_update()
104 # target_update()
105 to_update = [
105 to_update = [
106 (
106 (
107 "docs/man/ipython.1.gz",
107 "docs/man/ipython.1.gz",
108 ["docs/man/ipython.1"],
108 ["docs/man/ipython.1"],
109 "cd docs/man && python -m gzip --best ipython.1",
109 "cd docs/man && python -m gzip --best ipython.1",
110 ),
110 ),
111 ]
111 ]
112
112
113
113
114 [ target_update(*t) for t in to_update ]
114 [ target_update(*t) for t in to_update ]
115
115
116 #---------------------------------------------------------------------------
116 #---------------------------------------------------------------------------
117 # Find all the packages, package data, and data_files
117 # Find all the packages, package data, and data_files
118 #---------------------------------------------------------------------------
118 #---------------------------------------------------------------------------
119
119
120 data_files = find_data_files()
120 data_files = find_data_files()
121
121
122 setup_args['data_files'] = data_files
122 setup_args['data_files'] = data_files
123
123
124 #---------------------------------------------------------------------------
124 #---------------------------------------------------------------------------
125 # custom distutils commands
125 # custom distutils commands
126 #---------------------------------------------------------------------------
126 #---------------------------------------------------------------------------
127 # imports here, so they are after setuptools import if there was one
127 # imports here, so they are after setuptools import if there was one
128 from setuptools.command.sdist import sdist
128 from setuptools.command.sdist import sdist
129
129
130 setup_args['cmdclass'] = {
130 setup_args['cmdclass'] = {
131 'build_py': \
131 'build_py': \
132 check_package_data_first(git_prebuild('IPython')),
132 check_package_data_first(git_prebuild('IPython')),
133 'sdist' : git_prebuild('IPython', sdist),
133 'sdist' : git_prebuild('IPython', sdist),
134 'symlink': install_symlinked,
134 'symlink': install_symlinked,
135 'install_lib_symlink': install_lib_symlink,
135 'install_lib_symlink': install_lib_symlink,
136 'install_scripts_sym': install_scripts_for_symlink,
136 'install_scripts_sym': install_scripts_for_symlink,
137 'unsymlink': unsymlink,
137 'unsymlink': unsymlink,
138 }
138 }
139
139
140
140
141 #---------------------------------------------------------------------------
141 #---------------------------------------------------------------------------
142 # Handle scripts, dependencies, and setuptools specific things
142 # Handle scripts, dependencies, and setuptools specific things
143 #---------------------------------------------------------------------------
143 #---------------------------------------------------------------------------
144
144
145 # setuptools requirements
145 # setuptools requirements
146
146
147 extras_require = dict(
147 extras_require = dict(
148 parallel=["ipyparallel"],
148 parallel=["ipyparallel"],
149 qtconsole=["qtconsole"],
149 qtconsole=["qtconsole"],
150 doc=["Sphinx>=1.3"],
150 doc=["Sphinx>=1.3"],
151 test=[
151 test=[
152 "pytest",
152 "pytest",
153 "pytest-asyncio",
153 "testpath",
154 "testpath",
154 "pygments",
155 "pygments",
155 ],
156 ],
156 test_extra=[
157 test_extra=[
157 "pytest",
158 "pytest",
158 "testpath",
159 "testpath",
159 "curio",
160 "curio",
160 "matplotlib!=3.2.0",
161 "matplotlib!=3.2.0",
161 "nbformat",
162 "nbformat",
162 "numpy>=1.17",
163 "numpy>=1.17",
163 "pandas",
164 "pandas",
164 "pygments",
165 "pygments",
165 "trio",
166 "trio",
166 ],
167 ],
167 terminal=[],
168 terminal=[],
168 kernel=["ipykernel"],
169 kernel=["ipykernel"],
169 nbformat=["nbformat"],
170 nbformat=["nbformat"],
170 notebook=["notebook", "ipywidgets"],
171 notebook=["notebook", "ipywidgets"],
171 nbconvert=["nbconvert"],
172 nbconvert=["nbconvert"],
172 )
173 )
173
174
174 everything = set(chain.from_iterable(extras_require.values()))
175 everything = set(chain.from_iterable(extras_require.values()))
175 extras_require['all'] = list(sorted(everything))
176 extras_require['all'] = list(sorted(everything))
176
177
177 setup_args["extras_require"] = extras_require
178 setup_args["extras_require"] = extras_require
178
179
179 #---------------------------------------------------------------------------
180 #---------------------------------------------------------------------------
180 # Do the actual setup now
181 # Do the actual setup now
181 #---------------------------------------------------------------------------
182 #---------------------------------------------------------------------------
182
183
183 if __name__ == "__main__":
184 if __name__ == "__main__":
184 setup(**setup_args)
185 setup(**setup_args)
General Comments 0
You need to be logged in to leave comments. Login now