test_autoreload.py
712 lines
| 22.7 KiB
| text/x-python
|
PythonLexer
Fernando Perez
|
r6951 | """Tests for autoreload extension. | ||
""" | ||||
M Bussonnier
|
r28782 | |||
Spas Kalaydzhisyki
|
r26242 | # ----------------------------------------------------------------------------- | ||
Fernando Perez
|
r6951 | # Copyright (c) 2012 IPython Development Team. | ||
# | ||||
# Distributed under the terms of the Modified BSD License. | ||||
# | ||||
# The full license is in the file COPYING.txt, distributed with this software. | ||||
Spas Kalaydzhisyki
|
r26242 | # ----------------------------------------------------------------------------- | ||
Fernando Perez
|
r6951 | |||
Spas Kalaydzhisyki
|
r26242 | # ----------------------------------------------------------------------------- | ||
Fernando Perez
|
r6951 | # Imports | ||
Spas Kalaydzhisyki
|
r26242 | # ----------------------------------------------------------------------------- | ||
Fernando Perez
|
r6951 | |||
Pauli Virtanen
|
r4842 | import os | ||
Nikita Kniazev
|
r27235 | import platform | ||
import pytest | ||||
Pauli Virtanen
|
r4842 | import sys | ||
import tempfile | ||||
Matthias Bussonnier
|
r23397 | import textwrap | ||
Pauli Virtanen
|
r4842 | import shutil | ||
import random | ||||
import time | ||||
Matthias Bussonnier
|
r28501 | import traceback | ||
Srinivas Reddy Thatiparthy
|
r23079 | from io import StringIO | ||
Emilio Graff
|
r27805 | from dataclasses import dataclass | ||
Pauli Virtanen
|
r4842 | |||
Thomas Kluyver
|
r4904 | import IPython.testing.tools as tt | ||
Pauli Virtanen
|
r4842 | |||
Matthias Bussonnier
|
r25113 | from unittest import TestCase | ||
Brian Granger
|
r8197 | from IPython.extensions.autoreload import AutoreloadMagics | ||
Thomas Kluyver
|
r15607 | from IPython.core.events import EventManager, pre_run_cell | ||
sleeping
|
r27332 | from IPython.testing.decorators import skipif_not_numpy | ||
Matthias Bussonnier
|
r28501 | from IPython.core.interactiveshell import ExecutionInfo | ||
Pauli Virtanen
|
r4842 | |||
Nikita Kniazev
|
r27235 | if platform.python_implementation() == "PyPy": | ||
pytest.skip( | ||||
luz paz
|
r27520 | "Current autoreload implementation is extremely slow on PyPy", | ||
Nikita Kniazev
|
r27235 | allow_module_level=True, | ||
) | ||||
Spas Kalaydzhisyki
|
r26242 | # ----------------------------------------------------------------------------- | ||
Pauli Virtanen
|
r4842 | # Test fixture | ||
Spas Kalaydzhisyki
|
r26242 | # ----------------------------------------------------------------------------- | ||
Pauli Virtanen
|
r4842 | |||
Fernando Perez
|
r6951 | noop = lambda *a, **kw: None | ||
Matthias BUSSONNIER
|
r13237 | |||
Spas Kalaydzhisyki
|
r26242 | class FakeShell: | ||
Pauli Virtanen
|
r4842 | def __init__(self): | ||
self.ns = {} | ||||
Matthias Bussonnier
|
r25090 | self.user_ns = self.ns | ||
Matthias Bussonnier
|
r24976 | self.user_ns_hidden = {} | ||
Spas Kalaydzhisyki
|
r26242 | self.events = EventManager(self, {"pre_run_cell", pre_run_cell}) | ||
Brian Granger
|
r8197 | self.auto_magics = AutoreloadMagics(shell=self) | ||
Spas Kalaydzhisyki
|
r26242 | self.events.register("pre_run_cell", self.auto_magics.pre_run_cell) | ||
Fernando Perez
|
r6951 | |||
register_magics = set_hook = noop | ||||
Pauli Virtanen
|
r4842 | |||
Matthias Bussonnier
|
r28501 | def showtraceback( | ||
self, | ||||
exc_tuple=None, | ||||
filename=None, | ||||
tb_offset=None, | ||||
exception_only=False, | ||||
running_compiled_code=False, | ||||
): | ||||
traceback.print_exc() | ||||
Pauli Virtanen
|
r4842 | def run_code(self, code): | ||
Matthias Bussonnier
|
r28501 | self.events.trigger( | ||
"pre_run_cell", | ||||
ExecutionInfo( | ||||
raw_cell="", | ||||
store_history=False, | ||||
silent=False, | ||||
shell_futures=False, | ||||
cell_id=None, | ||||
), | ||||
) | ||||
Matthias Bussonnier
|
r25090 | exec(code, self.user_ns) | ||
Thomas Kluyver
|
r15682 | self.auto_magics.post_execute_hook() | ||
Pauli Virtanen
|
r4842 | |||
def push(self, items): | ||||
self.ns.update(items) | ||||
def magic_autoreload(self, parameter): | ||||
Brian Granger
|
r8197 | self.auto_magics.autoreload(parameter) | ||
Pauli Virtanen
|
r4842 | |||
def magic_aimport(self, parameter, stream=None): | ||||
Brian Granger
|
r8197 | self.auto_magics.aimport(parameter, stream=stream) | ||
Thomas Kluyver
|
r15682 | self.auto_magics.post_execute_hook() | ||
Pauli Virtanen
|
r4842 | |||
Emilio Graff
|
r27934 | |||
Matthias Bussonnier
|
r25113 | class Fixture(TestCase): | ||
Pauli Virtanen
|
r4842 | """Fixture for creating test module files""" | ||
test_dir = None | ||||
old_sys_path = None | ||||
filename_chars = "abcdefghijklmopqrstuvwxyz0123456789" | ||||
def setUp(self): | ||||
self.test_dir = tempfile.mkdtemp() | ||||
self.old_sys_path = list(sys.path) | ||||
sys.path.insert(0, self.test_dir) | ||||
self.shell = FakeShell() | ||||
def tearDown(self): | ||||
shutil.rmtree(self.test_dir) | ||||
sys.path = self.old_sys_path | ||||
self.test_dir = None | ||||
self.old_sys_path = None | ||||
self.shell = None | ||||
def get_module(self): | ||||
Spas Kalaydzhisyki
|
r26242 | module_name = "tmpmod_" + "".join(random.sample(self.filename_chars, 20)) | ||
Pauli Virtanen
|
r4842 | if module_name in sys.modules: | ||
del sys.modules[module_name] | ||||
file_name = os.path.join(self.test_dir, module_name + ".py") | ||||
return module_name, file_name | ||||
def write_file(self, filename, content): | ||||
""" | ||||
Write a file, and force a timestamp difference of at least one second | ||||
Notes | ||||
----- | ||||
Python's .pyc files record the timestamp of their compilation | ||||
with a time resolution of one second. | ||||
Therefore, we need to force a timestamp difference between .py | ||||
and .pyc, without having the .py file be timestamped in the | ||||
future, and without changing the timestamp of the .pyc file | ||||
(because that is stored in the file). The only reliable way | ||||
to achieve this seems to be to sleep. | ||||
""" | ||||
Matthias Bussonnier
|
r25090 | content = textwrap.dedent(content) | ||
Pauli Virtanen
|
r4842 | # Sleep one second + eps | ||
time.sleep(1.05) | ||||
# Write | ||||
gousaiyang
|
r27495 | with open(filename, "w", encoding="utf-8") as f: | ||
Pauli Virtanen
|
r4842 | f.write(content) | ||
def new_module(self, code): | ||||
Matthias Bussonnier
|
r25090 | code = textwrap.dedent(code) | ||
Pauli Virtanen
|
r4842 | mod_name, mod_fn = self.get_module() | ||
gousaiyang
|
r27495 | with open(mod_fn, "w", encoding="utf-8") as f: | ||
Pauli Virtanen
|
r4842 | f.write(code) | ||
return mod_name, mod_fn | ||||
Spas Kalaydzhisyki
|
r26242 | |||
# ----------------------------------------------------------------------------- | ||||
Pauli Virtanen
|
r4842 | # Test automatic reloading | ||
Spas Kalaydzhisyki
|
r26242 | # ----------------------------------------------------------------------------- | ||
Pauli Virtanen
|
r4842 | |||
Matthias Bussonnier
|
r25090 | def pickle_get_current_class(obj): | ||
""" | ||||
Original issue comes from pickle; hence the name. | ||||
""" | ||||
name = obj.__class__.__name__ | ||||
module_name = getattr(obj, "__module__", None) | ||||
obj2 = sys.modules[module_name] | ||||
for subpath in name.split("."): | ||||
obj2 = getattr(obj2, subpath) | ||||
return obj2 | ||||
Matthias Bussonnier
|
r23397 | |||
Spas Kalaydzhisyki
|
r26242 | class TestAutoreload(Fixture): | ||
Matthias Bussonnier
|
r23397 | def test_reload_enums(self): | ||
Spas Kalaydzhisyki
|
r26242 | mod_name, mod_fn = self.new_module( | ||
textwrap.dedent( | ||||
""" | ||||
Matthias Bussonnier
|
r23397 | from enum import Enum | ||
class MyEnum(Enum): | ||||
A = 'A' | ||||
B = 'B' | ||||
Spas Kalaydzhisyki
|
r26242 | """ | ||
) | ||||
) | ||||
Matthias Bussonnier
|
r23397 | self.shell.magic_autoreload("2") | ||
self.shell.magic_aimport(mod_name) | ||||
Spas Kalaydzhisyki
|
r26242 | self.write_file( | ||
mod_fn, | ||||
textwrap.dedent( | ||||
""" | ||||
Matthias Bussonnier
|
r23397 | from enum import Enum | ||
class MyEnum(Enum): | ||||
A = 'A' | ||||
B = 'B' | ||||
C = 'C' | ||||
Spas Kalaydzhisyki
|
r26242 | """ | ||
), | ||||
) | ||||
with tt.AssertNotPrints( | ||||
("[autoreload of %s failed:" % mod_name), channel="stderr" | ||||
): | ||||
Matthias Bussonnier
|
r23397 | self.shell.run_code("pass") # trigger another reload | ||
Matthias Bussonnier
|
r25090 | def test_reload_class_type(self): | ||
self.shell.magic_autoreload("2") | ||||
mod_name, mod_fn = self.new_module( | ||||
""" | ||||
class Test(): | ||||
def meth(self): | ||||
return "old" | ||||
""" | ||||
) | ||||
assert "test" not in self.shell.ns | ||||
assert "result" not in self.shell.ns | ||||
self.shell.run_code("from %s import Test" % mod_name) | ||||
self.shell.run_code("test = Test()") | ||||
self.write_file( | ||||
mod_fn, | ||||
""" | ||||
class Test(): | ||||
def meth(self): | ||||
return "new" | ||||
""", | ||||
) | ||||
test_object = self.shell.ns["test"] | ||||
# important to trigger autoreload logic ! | ||||
self.shell.run_code("pass") | ||||
test_class = pickle_get_current_class(test_object) | ||||
assert isinstance(test_object, test_class) | ||||
# extra check. | ||||
self.shell.run_code("import pickle") | ||||
self.shell.run_code("p = pickle.dumps(test)") | ||||
Matthias Bussonnier
|
r24515 | def test_reload_class_attributes(self): | ||
self.shell.magic_autoreload("2") | ||||
Spas Kalaydzhisyki
|
r26242 | mod_name, mod_fn = self.new_module( | ||
textwrap.dedent( | ||||
""" | ||||
Matthias Bussonnier
|
r24515 | class MyClass: | ||
def __init__(self, a=10): | ||||
self.a = a | ||||
self.b = 22 | ||||
# self.toto = 33 | ||||
def square(self): | ||||
print('compute square') | ||||
return self.a*self.a | ||||
Matthias Bussonnier
|
r24516 | """ | ||
) | ||||
) | ||||
Matthias Bussonnier
|
r24515 | self.shell.run_code("from %s import MyClass" % mod_name) | ||
Matthias Bussonnier
|
r24516 | self.shell.run_code("first = MyClass(5)") | ||
self.shell.run_code("first.square()") | ||||
Samuel Gaist
|
r26910 | with self.assertRaises(AttributeError): | ||
Matthias Bussonnier
|
r24516 | self.shell.run_code("first.cube()") | ||
Samuel Gaist
|
r26910 | with self.assertRaises(AttributeError): | ||
Matthias Bussonnier
|
r24516 | self.shell.run_code("first.power(5)") | ||
self.shell.run_code("first.b") | ||||
Samuel Gaist
|
r26910 | with self.assertRaises(AttributeError): | ||
Matthias Bussonnier
|
r24516 | self.shell.run_code("first.toto") | ||
Matthias Bussonnier
|
r24515 | |||
Matthias Bussonnier
|
r24516 | # remove square, add power | ||
Matthias Bussonnier
|
r24515 | |||
Matthias Bussonnier
|
r24516 | self.write_file( | ||
mod_fn, | ||||
textwrap.dedent( | ||||
""" | ||||
Matthias Bussonnier
|
r24515 | class MyClass: | ||
def __init__(self, a=10): | ||||
self.a = a | ||||
self.b = 11 | ||||
def power(self, p): | ||||
print('compute power '+str(p)) | ||||
return self.a**p | ||||
Matthias Bussonnier
|
r24516 | """ | ||
), | ||||
) | ||||
self.shell.run_code("second = MyClass(5)") | ||||
Spas Kalaydzhisyki
|
r26242 | for object_name in {"first", "second"}: | ||
Spas Kalaydzhisyki
|
r26240 | self.shell.run_code(f"{object_name}.power(5)") | ||
Samuel Gaist
|
r26910 | with self.assertRaises(AttributeError): | ||
Spas Kalaydzhisyki
|
r26240 | self.shell.run_code(f"{object_name}.cube()") | ||
Samuel Gaist
|
r26910 | with self.assertRaises(AttributeError): | ||
Spas Kalaydzhisyki
|
r26240 | self.shell.run_code(f"{object_name}.square()") | ||
self.shell.run_code(f"{object_name}.b") | ||||
self.shell.run_code(f"{object_name}.a") | ||||
Samuel Gaist
|
r26910 | with self.assertRaises(AttributeError): | ||
Spas Kalaydzhisyki
|
r26240 | self.shell.run_code(f"{object_name}.toto") | ||
Matthias Bussonnier
|
r23397 | |||
sleeping
|
r27332 | @skipif_not_numpy | ||
sleeping
|
r27331 | def test_comparing_numpy_structures(self): | ||
self.shell.magic_autoreload("2") | ||||
Matthias Bussonnier
|
r28501 | self.shell.run_code("1+1") | ||
sleeping
|
r27331 | mod_name, mod_fn = self.new_module( | ||
textwrap.dedent( | ||||
""" | ||||
import numpy as np | ||||
class MyClass: | ||||
a = (np.array((.1, .2)), | ||||
np.array((.2, .3))) | ||||
""" | ||||
) | ||||
) | ||||
self.shell.run_code("from %s import MyClass" % mod_name) | ||||
self.shell.run_code("first = MyClass()") | ||||
# change property `a` | ||||
self.write_file( | ||||
mod_fn, | ||||
textwrap.dedent( | ||||
""" | ||||
import numpy as np | ||||
class MyClass: | ||||
a = (np.array((.3, .4)), | ||||
np.array((.5, .6))) | ||||
""" | ||||
), | ||||
) | ||||
with tt.AssertNotPrints( | ||||
("[autoreload of %s failed:" % mod_name), channel="stderr" | ||||
): | ||||
self.shell.run_code("pass") # trigger another reload | ||||
Spas Kalaydzhisyki
|
r26238 | def test_autoload_newly_added_objects(self): | ||
Emilio Graff
|
r27804 | # All of these fail with %autoreload 2 | ||
Spas Kalaydzhisyki
|
r26238 | self.shell.magic_autoreload("3") | ||
mod_code = """ | ||||
def func1(): pass | ||||
""" | ||||
mod_name, mod_fn = self.new_module(textwrap.dedent(mod_code)) | ||||
self.shell.run_code(f"from {mod_name} import *") | ||||
self.shell.run_code("func1()") | ||||
Samuel Gaist
|
r26910 | with self.assertRaises(NameError): | ||
Spas Kalaydzhisyki
|
r26242 | self.shell.run_code("func2()") | ||
Samuel Gaist
|
r26910 | with self.assertRaises(NameError): | ||
Spas Kalaydzhisyki
|
r26242 | self.shell.run_code("t = Test()") | ||
Samuel Gaist
|
r26910 | with self.assertRaises(NameError): | ||
Spas Kalaydzhisyki
|
r26242 | self.shell.run_code("number") | ||
Spas Kalaydzhisyki
|
r26238 | |||
# ----------- TEST NEW OBJ LOADED -------------------------- | ||||
new_code = """ | ||||
def func1(): pass | ||||
def func2(): pass | ||||
class Test: pass | ||||
number = 0 | ||||
from enum import Enum | ||||
class TestEnum(Enum): | ||||
A = 'a' | ||||
""" | ||||
self.write_file(mod_fn, textwrap.dedent(new_code)) | ||||
# test function now exists in shell's namespace namespace | ||||
self.shell.run_code("func2()") | ||||
# test function now exists in module's dict | ||||
self.shell.run_code(f"import sys; sys.modules['{mod_name}'].func2()") | ||||
# test class now exists | ||||
self.shell.run_code("t = Test()") | ||||
# test global built-in var now exists | ||||
Spas Kalaydzhisyki
|
r26242 | self.shell.run_code("number") | ||
Dimitri Papadopoulos
|
r26875 | # test the enumerations gets loaded successfully | ||
Spas Kalaydzhisyki
|
r26238 | self.shell.run_code("TestEnum.A") | ||
# ----------- TEST NEW OBJ CAN BE CHANGED -------------------- | ||||
new_code = """ | ||||
def func1(): return 'changed' | ||||
def func2(): return 'changed' | ||||
class Test: | ||||
def new_func(self): | ||||
return 'changed' | ||||
number = 1 | ||||
from enum import Enum | ||||
class TestEnum(Enum): | ||||
A = 'a' | ||||
B = 'added' | ||||
""" | ||||
self.write_file(mod_fn, textwrap.dedent(new_code)) | ||||
self.shell.run_code("assert func1() == 'changed'") | ||||
self.shell.run_code("assert func2() == 'changed'") | ||||
self.shell.run_code("t = Test(); assert t.new_func() == 'changed'") | ||||
self.shell.run_code("assert number == 1") | ||||
Matthias Bussonnier
|
r27975 | if sys.version_info < (3, 12): | ||
self.shell.run_code("assert TestEnum.B.value == 'added'") | ||||
Spas Kalaydzhisyki
|
r26238 | |||
# ----------- TEST IMPORT FROM MODULE -------------------------- | ||||
Spas Kalaydzhisyki
|
r26242 | new_mod_code = """ | ||
Spas Kalaydzhisyki
|
r26238 | from enum import Enum | ||
class Ext(Enum): | ||||
A = 'ext' | ||||
def ext_func(): | ||||
return 'ext' | ||||
class ExtTest: | ||||
def meth(self): | ||||
return 'ext' | ||||
ext_int = 2 | ||||
Spas Kalaydzhisyki
|
r26242 | """ | ||
Spas Kalaydzhisyki
|
r26238 | new_mod_name, new_mod_fn = self.new_module(textwrap.dedent(new_mod_code)) | ||
Spas Kalaydzhisyki
|
r26242 | current_mod_code = f""" | ||
Spas Kalaydzhisyki
|
r26238 | from {new_mod_name} import * | ||
Spas Kalaydzhisyki
|
r26242 | """ | ||
Spas Kalaydzhisyki
|
r26238 | self.write_file(mod_fn, textwrap.dedent(current_mod_code)) | ||
self.shell.run_code("assert Ext.A.value == 'ext'") | ||||
self.shell.run_code("assert ext_func() == 'ext'") | ||||
self.shell.run_code("t = ExtTest(); assert t.meth() == 'ext'") | ||||
self.shell.run_code("assert ext_int == 2") | ||||
Emilio Graff
|
r27805 | def test_verbose_names(self): | ||
# Asserts correspondense between original mode names and their verbose equivalents. | ||||
@dataclass | ||||
class AutoreloadSettings: | ||||
check_all: bool | ||||
enabled: bool | ||||
autoload_obj: bool | ||||
def gather_settings(mode): | ||||
self.shell.magic_autoreload(mode) | ||||
module_reloader = self.shell.auto_magics._reloader | ||||
Emilio Graff
|
r27809 | return AutoreloadSettings( | ||
module_reloader.check_all, | ||||
module_reloader.enabled, | ||||
module_reloader.autoload_obj, | ||||
) | ||||
Emilio Graff
|
r27805 | assert gather_settings("0") == gather_settings("off") | ||
assert gather_settings("0") == gather_settings("OFF") # Case insensitive | ||||
assert gather_settings("1") == gather_settings("explicit") | ||||
assert gather_settings("2") == gather_settings("all") | ||||
assert gather_settings("3") == gather_settings("complete") | ||||
# And an invalid mode name raises an exception. | ||||
with self.assertRaises(ValueError): | ||||
Emilio Graff
|
r27809 | self.shell.magic_autoreload("4") | ||
Emilio Graff
|
r27805 | |||
Emilio Graff
|
r27806 | def test_aimport_parsing(self): | ||
# Modules can be included or excluded all in one line. | ||||
module_reloader = self.shell.auto_magics._reloader | ||||
Emilio Graff
|
r27809 | self.shell.magic_aimport("os") # import and mark `os` for auto-reload. | ||
assert module_reloader.modules["os"] is True | ||||
assert "os" not in module_reloader.skip_modules.keys() | ||||
self.shell.magic_aimport("-math") # forbid autoreloading of `math` | ||||
assert module_reloader.skip_modules["math"] is True | ||||
assert "math" not in module_reloader.modules.keys() | ||||
self.shell.magic_aimport( | ||||
"-os, math" | ||||
) # Can do this all in one line; wasn't possible before. | ||||
assert module_reloader.modules["math"] is True | ||||
assert "math" not in module_reloader.skip_modules.keys() | ||||
assert module_reloader.skip_modules["os"] is True | ||||
assert "os" not in module_reloader.modules.keys() | ||||
Emilio Graff
|
r27806 | |||
Emilio Graff
|
r27931 | def test_autoreload_output(self): | ||
Emilio Graff
|
r27807 | self.shell.magic_autoreload("complete") | ||
mod_code = """ | ||||
def func1(): pass | ||||
""" | ||||
mod_name, mod_fn = self.new_module(mod_code) | ||||
self.shell.run_code(f"import {mod_name}") | ||||
Emilio Graff
|
r27931 | with tt.AssertPrints("", channel="stdout"): # no output; this is default | ||
Emilio Graff
|
r27807 | self.shell.run_code("pass") | ||
Emilio Graff
|
r27931 | self.shell.magic_autoreload("complete --print") | ||
Emilio Graff
|
r27807 | self.write_file(mod_fn, mod_code) # "modify" the module | ||
Emilio Graff
|
r27934 | with tt.AssertPrints( | ||
f"Reloading '{mod_name}'.", channel="stdout" | ||||
): # see something printed out | ||||
Emilio Graff
|
r27807 | self.shell.run_code("pass") | ||
Emilio Graff
|
r27931 | self.shell.magic_autoreload("complete -p") | ||
Emilio Graff
|
r27807 | self.write_file(mod_fn, mod_code) # "modify" the module | ||
Emilio Graff
|
r27934 | with tt.AssertPrints( | ||
f"Reloading '{mod_name}'.", channel="stdout" | ||||
): # see something printed out | ||||
Emilio Graff
|
r27807 | self.shell.run_code("pass") | ||
Emilio Graff
|
r27931 | self.shell.magic_autoreload("complete --print --log") | ||
self.write_file(mod_fn, mod_code) # "modify" the module | ||||
Emilio Graff
|
r27934 | with tt.AssertPrints( | ||
f"Reloading '{mod_name}'.", channel="stdout" | ||||
): # see something printed out | ||||
Emilio Graff
|
r27931 | self.shell.run_code("pass") | ||
Emilio Graff
|
r27807 | |||
Emilio Graff
|
r27932 | self.shell.magic_autoreload("complete --print --log") | ||
self.write_file(mod_fn, mod_code) # "modify" the module | ||||
Emilio Graff
|
r27934 | with self.assertLogs(logger="autoreload") as lo: # see something printed out | ||
Emilio Graff
|
r27932 | self.shell.run_code("pass") | ||
Emilio Graff
|
r27934 | assert lo.output == [f"INFO:autoreload:Reloading '{mod_name}'."] | ||
Emilio Graff
|
r27932 | |||
self.shell.magic_autoreload("complete -l") | ||||
self.write_file(mod_fn, mod_code) # "modify" the module | ||||
Emilio Graff
|
r27934 | with self.assertLogs(logger="autoreload") as lo: # see something printed out | ||
Emilio Graff
|
r27932 | self.shell.run_code("pass") | ||
Emilio Graff
|
r27934 | assert lo.output == [f"INFO:autoreload:Reloading '{mod_name}'."] | ||
Emilio Graff
|
r27807 | |||
Pauli Virtanen
|
r4842 | def _check_smoketest(self, use_aimport=True): | ||
""" | ||||
Functional test for the automatic reloader using either | ||||
'%autoreload 1' or '%autoreload 2' | ||||
""" | ||||
Spas Kalaydzhisyki
|
r26242 | mod_name, mod_fn = self.new_module( | ||
""" | ||||
Pauli Virtanen
|
r4842 | x = 9 | ||
z = 123 # this item will be deleted | ||||
def foo(y): | ||||
return y + 3 | ||||
class Baz(object): | ||||
def __init__(self, x): | ||||
self.x = x | ||||
def bar(self, y): | ||||
return self.x + y | ||||
@property | ||||
def quux(self): | ||||
return 42 | ||||
def zzz(self): | ||||
'''This method will be deleted below''' | ||||
return 99 | ||||
class Bar: # old-style class: weakref doesn't work for it on Python < 2.7 | ||||
def foo(self): | ||||
return 1 | ||||
Spas Kalaydzhisyki
|
r26242 | """ | ||
) | ||||
Pauli Virtanen
|
r4842 | |||
# | ||||
# Import module, and mark for reloading | ||||
# | ||||
if use_aimport: | ||||
self.shell.magic_autoreload("1") | ||||
self.shell.magic_aimport(mod_name) | ||||
stream = StringIO() | ||||
self.shell.magic_aimport("", stream=stream) | ||||
Samuel Gaist
|
r26910 | self.assertIn(("Modules to reload:\n%s" % mod_name), stream.getvalue()) | ||
Pauli Virtanen
|
r4842 | |||
Samuel Gaist
|
r26910 | with self.assertRaises(ImportError): | ||
Thomas Kluyver
|
r15682 | self.shell.magic_aimport("tmpmod_as318989e89ds") | ||
Pauli Virtanen
|
r4842 | else: | ||
self.shell.magic_autoreload("2") | ||||
self.shell.run_code("import %s" % mod_name) | ||||
stream = StringIO() | ||||
self.shell.magic_aimport("", stream=stream) | ||||
Samuel Gaist
|
r26910 | self.assertTrue( | ||
Spas Kalaydzhisyki
|
r26242 | "Modules to reload:\nall-except-skipped" in stream.getvalue() | ||
) | ||||
Samuel Gaist
|
r26910 | self.assertIn(mod_name, self.shell.ns) | ||
Pauli Virtanen
|
r4842 | |||
mod = sys.modules[mod_name] | ||||
# | ||||
# Test module contents | ||||
# | ||||
old_foo = mod.foo | ||||
old_obj = mod.Baz(9) | ||||
old_obj2 = mod.Bar() | ||||
def check_module_contents(): | ||||
Samuel Gaist
|
r26910 | self.assertEqual(mod.x, 9) | ||
self.assertEqual(mod.z, 123) | ||||
Pauli Virtanen
|
r4842 | |||
Samuel Gaist
|
r26910 | self.assertEqual(old_foo(0), 3) | ||
self.assertEqual(mod.foo(0), 3) | ||||
Pauli Virtanen
|
r4842 | |||
obj = mod.Baz(9) | ||||
Samuel Gaist
|
r26910 | self.assertEqual(old_obj.bar(1), 10) | ||
self.assertEqual(obj.bar(1), 10) | ||||
self.assertEqual(obj.quux, 42) | ||||
self.assertEqual(obj.zzz(), 99) | ||||
Pauli Virtanen
|
r4842 | |||
obj2 = mod.Bar() | ||||
Samuel Gaist
|
r26910 | self.assertEqual(old_obj2.foo(), 1) | ||
self.assertEqual(obj2.foo(), 1) | ||||
Pauli Virtanen
|
r4842 | |||
check_module_contents() | ||||
# | ||||
# Simulate a failed reload: no reload should occur and exactly | ||||
# one error message should be printed | ||||
# | ||||
Spas Kalaydzhisyki
|
r26242 | self.write_file( | ||
mod_fn, | ||||
""" | ||||
Pauli Virtanen
|
r4842 | a syntax error | ||
Spas Kalaydzhisyki
|
r26242 | """, | ||
) | ||||
Pauli Virtanen
|
r4842 | |||
Spas Kalaydzhisyki
|
r26242 | with tt.AssertPrints( | ||
("[autoreload of %s failed:" % mod_name), channel="stderr" | ||||
): | ||||
self.shell.run_code("pass") # trigger reload | ||||
with tt.AssertNotPrints( | ||||
("[autoreload of %s failed:" % mod_name), channel="stderr" | ||||
): | ||||
self.shell.run_code("pass") # trigger another reload | ||||
Thomas Kluyver
|
r4904 | check_module_contents() | ||
Pauli Virtanen
|
r4842 | |||
# | ||||
# Rewrite module (this time reload should succeed) | ||||
# | ||||
Spas Kalaydzhisyki
|
r26242 | self.write_file( | ||
mod_fn, | ||||
""" | ||||
Pauli Virtanen
|
r4842 | x = 10 | ||
def foo(y): | ||||
return y + 4 | ||||
class Baz(object): | ||||
def __init__(self, x): | ||||
self.x = x | ||||
def bar(self, y): | ||||
return self.x + y + 1 | ||||
@property | ||||
def quux(self): | ||||
return 43 | ||||
class Bar: # old-style class | ||||
def foo(self): | ||||
return 2 | ||||
Spas Kalaydzhisyki
|
r26242 | """, | ||
) | ||||
Pauli Virtanen
|
r4842 | |||
def check_module_contents(): | ||||
Samuel Gaist
|
r26910 | self.assertEqual(mod.x, 10) | ||
self.assertFalse(hasattr(mod, "z")) | ||||
Pauli Virtanen
|
r4842 | |||
Samuel Gaist
|
r26910 | self.assertEqual(old_foo(0), 4) # superreload magic! | ||
self.assertEqual(mod.foo(0), 4) | ||||
Pauli Virtanen
|
r4842 | |||
obj = mod.Baz(9) | ||||
Samuel Gaist
|
r26910 | self.assertEqual(old_obj.bar(1), 11) # superreload magic! | ||
self.assertEqual(obj.bar(1), 11) | ||||
Pauli Virtanen
|
r4842 | |||
Samuel Gaist
|
r26910 | self.assertEqual(old_obj.quux, 43) | ||
self.assertEqual(obj.quux, 43) | ||||
Pauli Virtanen
|
r4842 | |||
Samuel Gaist
|
r26910 | self.assertFalse(hasattr(old_obj, "zzz")) | ||
self.assertFalse(hasattr(obj, "zzz")) | ||||
Pauli Virtanen
|
r4842 | |||
obj2 = mod.Bar() | ||||
Samuel Gaist
|
r26910 | self.assertEqual(old_obj2.foo(), 2) | ||
self.assertEqual(obj2.foo(), 2) | ||||
Pauli Virtanen
|
r4842 | |||
Spas Kalaydzhisyki
|
r26242 | self.shell.run_code("pass") # trigger reload | ||
Pauli Virtanen
|
r4842 | check_module_contents() | ||
# | ||||
# Another failure case: deleted file (shouldn't reload) | ||||
# | ||||
os.unlink(mod_fn) | ||||
Spas Kalaydzhisyki
|
r26242 | self.shell.run_code("pass") # trigger reload | ||
Pauli Virtanen
|
r4842 | check_module_contents() | ||
# | ||||
# Disable autoreload and rewrite module: no reload should occur | ||||
# | ||||
if use_aimport: | ||||
self.shell.magic_aimport("-" + mod_name) | ||||
stream = StringIO() | ||||
self.shell.magic_aimport("", stream=stream) | ||||
Samuel Gaist
|
r26910 | self.assertTrue(("Modules to skip:\n%s" % mod_name) in stream.getvalue()) | ||
Pauli Virtanen
|
r4842 | |||
# This should succeed, although no such module exists | ||||
self.shell.magic_aimport("-tmpmod_as318989e89ds") | ||||
else: | ||||
self.shell.magic_autoreload("0") | ||||
Spas Kalaydzhisyki
|
r26242 | self.write_file( | ||
mod_fn, | ||||
""" | ||||
Pauli Virtanen
|
r4842 | x = -99 | ||
Spas Kalaydzhisyki
|
r26242 | """, | ||
) | ||||
Pauli Virtanen
|
r4842 | |||
Spas Kalaydzhisyki
|
r26242 | self.shell.run_code("pass") # trigger reload | ||
Pauli Virtanen
|
r4842 | self.shell.run_code("pass") | ||
check_module_contents() | ||||
# | ||||
# Re-enable autoreload: reload should now occur | ||||
# | ||||
if use_aimport: | ||||
self.shell.magic_aimport(mod_name) | ||||
else: | ||||
self.shell.magic_autoreload("") | ||||
Spas Kalaydzhisyki
|
r26242 | self.shell.run_code("pass") # trigger reload | ||
Samuel Gaist
|
r26910 | self.assertEqual(mod.x, -99) | ||
Pauli Virtanen
|
r4842 | |||
def test_smoketest_aimport(self): | ||||
self._check_smoketest(use_aimport=True) | ||||
def test_smoketest_autoreload(self): | ||||
self._check_smoketest(use_aimport=False) | ||||