test_autoreload.py
447 lines
| 13.7 KiB
| text/x-python
|
PythonLexer
Fernando Perez
|
r6951 | """Tests for autoreload extension. | ||
""" | ||||
#----------------------------------------------------------------------------- | ||||
# 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. | ||||
#----------------------------------------------------------------------------- | ||||
#----------------------------------------------------------------------------- | ||||
# Imports | ||||
#----------------------------------------------------------------------------- | ||||
Pauli Virtanen
|
r4842 | import os | ||
import sys | ||||
import tempfile | ||||
Matthias Bussonnier
|
r23397 | import textwrap | ||
Pauli Virtanen
|
r4842 | import shutil | ||
import random | ||||
import time | ||||
Srinivas Reddy Thatiparthy
|
r23079 | from io import StringIO | ||
Pauli Virtanen
|
r4842 | |||
import nose.tools as nt | ||||
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 | ||
Pauli Virtanen
|
r4842 | |||
#----------------------------------------------------------------------------- | ||||
# Test fixture | ||||
#----------------------------------------------------------------------------- | ||||
Fernando Perez
|
r6951 | noop = lambda *a, **kw: None | ||
Matthias Bussonnier
|
r24976 | class FakeShell: | ||
Matthias BUSSONNIER
|
r13237 | |||
Pauli Virtanen
|
r4842 | def __init__(self): | ||
self.ns = {} | ||||
Matthias Bussonnier
|
r25090 | self.user_ns = self.ns | ||
Matthias Bussonnier
|
r24976 | self.user_ns_hidden = {} | ||
Thomas Kluyver
|
r15607 | self.events = EventManager(self, {'pre_run_cell', pre_run_cell}) | ||
Brian Granger
|
r8197 | self.auto_magics = AutoreloadMagics(shell=self) | ||
Thomas Kluyver
|
r15607 | self.events.register('pre_run_cell', self.auto_magics.pre_run_cell) | ||
Fernando Perez
|
r6951 | |||
register_magics = set_hook = noop | ||||
Pauli Virtanen
|
r4842 | |||
def run_code(self, code): | ||||
Thomas Kluyver
|
r15607 | self.events.trigger('pre_run_cell') | ||
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 | |||
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): | ||||
module_name = "tmpmod_" + "".join(random.sample(self.filename_chars,20)) | ||||
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 | ||||
Mickaël Schoentgen
|
r24897 | with open(filename, 'w') 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() | ||
Mickaël Schoentgen
|
r24897 | with open(mod_fn, 'w') as f: | ||
Pauli Virtanen
|
r4842 | f.write(code) | ||
return mod_name, mod_fn | ||||
#----------------------------------------------------------------------------- | ||||
# Test automatic reloading | ||||
#----------------------------------------------------------------------------- | ||||
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 | ||||
Pauli Virtanen
|
r4842 | class TestAutoreload(Fixture): | ||
Matthias Bussonnier
|
r23397 | |||
def test_reload_enums(self): | ||||
mod_name, mod_fn = self.new_module(textwrap.dedent(""" | ||||
from enum import Enum | ||||
class MyEnum(Enum): | ||||
A = 'A' | ||||
B = 'B' | ||||
""")) | ||||
self.shell.magic_autoreload("2") | ||||
self.shell.magic_aimport(mod_name) | ||||
self.write_file(mod_fn, textwrap.dedent(""" | ||||
from enum import Enum | ||||
class MyEnum(Enum): | ||||
A = 'A' | ||||
B = 'B' | ||||
C = 'C' | ||||
""")) | ||||
with tt.AssertNotPrints(('[autoreload of %s failed:' % mod_name), channel='stderr'): | ||||
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") | ||||
mod_name, mod_fn = self.new_module(textwrap.dedent(""" | ||||
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()") | ||||
Matthias Bussonnier
|
r24515 | with nt.assert_raises(AttributeError): | ||
Matthias Bussonnier
|
r24516 | self.shell.run_code("first.cube()") | ||
Matthias Bussonnier
|
r24515 | with nt.assert_raises(AttributeError): | ||
Matthias Bussonnier
|
r24516 | self.shell.run_code("first.power(5)") | ||
self.shell.run_code("first.b") | ||||
Matthias Bussonnier
|
r24515 | with nt.assert_raises(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)") | ||||
for object_name in {'first', 'second'}: | ||||
self.shell.run_code("{object_name}.power(5)".format(object_name=object_name)) | ||||
with nt.assert_raises(AttributeError): | ||||
self.shell.run_code("{object_name}.cube()".format(object_name=object_name)) | ||||
with nt.assert_raises(AttributeError): | ||||
oscar6echo
|
r24520 | self.shell.run_code("{object_name}.square()".format(object_name=object_name)) | ||
Matthias Bussonnier
|
r24516 | self.shell.run_code("{object_name}.b".format(object_name=object_name)) | ||
self.shell.run_code("{object_name}.a".format(object_name=object_name)) | ||||
with nt.assert_raises(AttributeError): | ||||
self.shell.run_code("{object_name}.toto".format(object_name=object_name)) | ||||
Matthias Bussonnier
|
r23397 | |||
Pauli Virtanen
|
r4842 | def _check_smoketest(self, use_aimport=True): | ||
""" | ||||
Functional test for the automatic reloader using either | ||||
'%autoreload 1' or '%autoreload 2' | ||||
""" | ||||
mod_name, mod_fn = self.new_module(""" | ||||
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 | ||||
""") | ||||
# | ||||
# 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) | ||||
Thomas Kluyver
|
r15682 | nt.assert_in(("Modules to reload:\n%s" % mod_name), stream.getvalue()) | ||
Pauli Virtanen
|
r4842 | |||
Thomas Kluyver
|
r15682 | with nt.assert_raises(ImportError): | ||
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) | ||||
nt.assert_true("Modules to reload:\nall-except-skipped" in | ||||
stream.getvalue()) | ||||
Fernando Perez
|
r6947 | nt.assert_in(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(): | ||||
nt.assert_equal(mod.x, 9) | ||||
nt.assert_equal(mod.z, 123) | ||||
nt.assert_equal(old_foo(0), 3) | ||||
nt.assert_equal(mod.foo(0), 3) | ||||
obj = mod.Baz(9) | ||||
nt.assert_equal(old_obj.bar(1), 10) | ||||
nt.assert_equal(obj.bar(1), 10) | ||||
nt.assert_equal(obj.quux, 42) | ||||
nt.assert_equal(obj.zzz(), 99) | ||||
obj2 = mod.Bar() | ||||
nt.assert_equal(old_obj2.foo(), 1) | ||||
nt.assert_equal(obj2.foo(), 1) | ||||
check_module_contents() | ||||
# | ||||
# Simulate a failed reload: no reload should occur and exactly | ||||
# one error message should be printed | ||||
# | ||||
self.write_file(mod_fn, """ | ||||
a syntax error | ||||
""") | ||||
Thomas Kluyver
|
r4904 | with tt.AssertPrints(('[autoreload of %s failed:' % mod_name), channel='stderr'): | ||
Pauli Virtanen
|
r4842 | self.shell.run_code("pass") # trigger reload | ||
Thomas Kluyver
|
r4904 | with tt.AssertNotPrints(('[autoreload of %s failed:' % mod_name), channel='stderr'): | ||
Pauli Virtanen
|
r4842 | self.shell.run_code("pass") # trigger another reload | ||
Thomas Kluyver
|
r4904 | check_module_contents() | ||
Pauli Virtanen
|
r4842 | |||
# | ||||
# Rewrite module (this time reload should succeed) | ||||
# | ||||
self.write_file(mod_fn, """ | ||||
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 | ||||
""") | ||||
def check_module_contents(): | ||||
nt.assert_equal(mod.x, 10) | ||||
nt.assert_false(hasattr(mod, 'z')) | ||||
nt.assert_equal(old_foo(0), 4) # superreload magic! | ||||
nt.assert_equal(mod.foo(0), 4) | ||||
obj = mod.Baz(9) | ||||
nt.assert_equal(old_obj.bar(1), 11) # superreload magic! | ||||
nt.assert_equal(obj.bar(1), 11) | ||||
nt.assert_equal(old_obj.quux, 43) | ||||
nt.assert_equal(obj.quux, 43) | ||||
nt.assert_false(hasattr(old_obj, 'zzz')) | ||||
nt.assert_false(hasattr(obj, 'zzz')) | ||||
obj2 = mod.Bar() | ||||
nt.assert_equal(old_obj2.foo(), 2) | ||||
nt.assert_equal(obj2.foo(), 2) | ||||
self.shell.run_code("pass") # trigger reload | ||||
check_module_contents() | ||||
# | ||||
# Another failure case: deleted file (shouldn't reload) | ||||
# | ||||
os.unlink(mod_fn) | ||||
self.shell.run_code("pass") # trigger reload | ||||
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) | ||||
nt.assert_true(("Modules to skip:\n%s" % mod_name) in | ||||
stream.getvalue()) | ||||
# This should succeed, although no such module exists | ||||
self.shell.magic_aimport("-tmpmod_as318989e89ds") | ||||
else: | ||||
self.shell.magic_autoreload("0") | ||||
self.write_file(mod_fn, """ | ||||
x = -99 | ||||
""") | ||||
self.shell.run_code("pass") # trigger reload | ||||
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("") | ||||
self.shell.run_code("pass") # trigger reload | ||||
nt.assert_equal(mod.x, -99) | ||||
def test_smoketest_aimport(self): | ||||
self._check_smoketest(use_aimport=True) | ||||
def test_smoketest_autoreload(self): | ||||
self._check_smoketest(use_aimport=False) | ||||
Matthias Bussonnier
|
r24515 | |||
Matthias Bussonnier
|
r25090 | |||