"""ImportSpy
Keeps track of modules not imported directly by Webware for Python.
This module helps save the filepath of every module which is imported.
This is used by the `AutoReloadingAppServer` (see doc strings for more
information) to restart the server if any source files change.
Other than keeping track of the filepaths, the behaviour of this module
loader is identical to Python's default behaviour.
If the system supports FAM (file alteration monitor) and python-fam is
installed, then the need for reloading can be monitored very effectively
with the use of ImportSpy. Otherwise, ImportSpy will not have much benefit.
Note that ImportSpy is based on the new import hooks of Python 2.3 described in
PEP 302, falling back to the old ihooks module if new hooks are not available.
In some cases this may become problematic, when other templating systems are
used with Webware which are also using ihook support to load their templates,
or if they are using zipimports. Therefore, it is possible to suppress the use
of ImportSpy by setting `UseImportSpy` in AppServer.config to False.
"""
try:
from sys import path_hooks, path_importer_cache
except ImportError:
path_hooks = None
if path_hooks is not None:
from os.path import isdir
class ImportSpy(object):
"""New style import tracker."""
_imp = None
def __init__(self, path=None):
"""Create importer."""
assert self._imp
if path and isdir(path):
self.path = path
else:
raise ImportError
def find_module(self, fullname):
"""Replaces imp.find_module."""
try:
self.file, self.filename, self.info = self._imp.find_module(
fullname.split('.')[-1], [self.path])
except ImportError:
pass
else:
return self
def load_module(self, fullname):
"""Replaces imp.load_module."""
mod = self._imp.load_module(fullname, self.file, self.filename, self.info)
if mod:
mod.__loader__ = self
return mod
def activate(impManager):
"""Activate ImportSpy."""
assert not ImportSpy._imp
ImportSpy._imp = impManager
path_hooks.append(ImportSpy)
path_importer_cache.clear()
impManager.recordModules()
return 'new import hooks'
else:
import ihooks
class ImportSpy(ihooks.ModuleLoader):
"""Old style import tracker."""
_imp = None
def __init__(self):
"""Create import hook."""
assert self._imp
ihooks.ModuleLoader.__init__(self)
self._lock = self._imp._lock
imp = ihooks.ModuleImporter(loader=self)
ihooks.install(imp)
self._imp.recordModules()
def load_module(self, name, stuff):
"""Replaces imp.load_module."""
file, filename, info = stuff
try:
try:
self._lock.acquire()
mod = ihooks.ModuleLoader.load_module(self, name, stuff)
finally:
self._lock.release()
self._imp.recordModule(mod)
except:
self._imp.recordFile(filename)
raise
return mod
def activate(impManager):
"""Activate ImportSpy."""
assert not ImportSpy._imp
ImportSpy._imp = impManager
ImportSpy()
return 'ihooks'