Skip to content
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Next Next commit
Refactor filewatching and add tests.
Change API of filewatch.py, update curtsies repl to reflect changes.
  • Loading branch information
keyan committed Jan 17, 2015
commit 3a65ba9cc04bc2d6b53c2da9252336476b7aec3d
40 changes: 24 additions & 16 deletions bpython/curtsiesfrontend/filewatch.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,17 +19,20 @@ def __init__(self, paths, on_change):
self.observer = Observer()
self.old_dirs = defaultdict(set)
self.started = False
self.activated = False
for path in paths:
self.add_module(path)
self._add_module(path)

def reset(self):
self.dirs = defaultdict(set)
del self.modules_to_add_later[:]
self.old_dirs = defaultdict(set)
self.observer.unschedule_all()

def add_module(self, path):
"""Add a python module to track changes to"""
def _add_module(self, path):
"""Add a python module to track changes to

Can"""
path = os.path.abspath(path)
for suff in importcompletion.SUFFIXES:
if path.endswith(suff):
Expand All @@ -40,24 +43,36 @@ def add_module(self, path):
self.observer.schedule(self, dirname, recursive=False)
self.dirs[os.path.dirname(path)].add(path)

def add_module_later(self, path):
def _add_module_later(self, path):
self.modules_to_add_later.append(path)

def track_module(self, path):
"""
Begins tracking this if activated, or remembers to track later.
"""
if self.activated:
self._add_module(path)
else:
self._add_module_later(path)

def activate(self):
if self.activated:
raise ValueError("%r is already activated." % (self,))
if not self.started:
self.started = True
self.observer.start()
self.dirs = self.old_dirs
for dirname in self.dirs:
self.observer.schedule(self, dirname, recursive=False)
for module in self.modules_to_add_later:
self.add_module(module)
self._add_module(module)
del self.modules_to_add_later[:]
self.activated = True

def deactivate(self):
if not self.activated:
raise ValueError("%r is not activated." % (self,))
self.observer.unschedule_all()
self.old_dirs = self.dirs
self.dirs = defaultdict(set)
self.activated = False

def on_any_event(self, event):
dirpath = os.path.dirname(event.src_path)
Expand All @@ -66,12 +81,5 @@ def on_any_event(self, event):
self.on_change(files_modified=[event.src_path])

if __name__ == '__main__':
m = ModuleChangedEventHandler([])
m.add_module('./wdtest.py')
try:
while True:
time.sleep(1)
except KeyboardInterrupt:
m.observer.stop()
m.observer.join()
pass

10 changes: 2 additions & 8 deletions bpython/curtsiesfrontend/repl.py
Original file line number Diff line number Diff line change
Expand Up @@ -366,18 +366,12 @@ def new_import(name, globals={}, locals={}, fromlist=[], level=-1):
except:
if name in old_module_locations:
loc = old_module_locations[name]
if self.watching_files:
self.watcher.add_module(loc)
else:
self.watcher.add_module_later(loc)
self.watcher.track_module(loc)
raise
else:
if hasattr(m, "__file__"):
old_module_locations[name] = m.__file__
if self.watching_files:
self.watcher.add_module(m.__file__)
else:
self.watcher.add_module_later(m.__file__)
self.watcher.track_module(m.__file__)
return m
__builtins__['__import__'] = new_import

Expand Down
30 changes: 30 additions & 0 deletions bpython/test/test_filewatch.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
import mock
import os

try:
import unittest2 as unittest
except ImportError:
import unittest

from bpython.curtsiesfrontend.filewatch import ModuleChangedEventHandler

class TestModuleChangeEventHandler(unittest.TestCase):

def setUp(self):
self.module = ModuleChangedEventHandler([], 1)
self.module.observer = mock.Mock()

def test_create_module_handler(self):
self.assertIsInstance(self.module, ModuleChangedEventHandler)

def test_add_module(self):
self.module._add_module('something/test.py')
self.assertIn(os.path.abspath('something/test'),
self.module.dirs[os.path.abspath('something')])

def test_activate_throws_error_when_already_activated(self):
self.module.activated = True
with self.assertRaises(ValueError):
self.module.activate()