[Zope-Checkins] CVS: Zope/lib/python/App - RefreshFuncs.py:1.1.4.1
Shane Hathaway
shane@cvs.zope.org
Mon, 6 Aug 2001 11:06:35 -0400
Update of /cvs-repository/Zope/lib/python/App
In directory cvs.zope.org:/tmp/cvs-serv28114/lib/python/App
Modified Files:
Tag: Zope-2_4-branch
RefreshFuncs.py
Log Message:
Simplified the way product refresh happens. Instead of using the reload()
function, we just remove the appropriate modules from sys.modules. This
ensures that modules are loaded in the correct order.
=== Zope/lib/python/App/RefreshFuncs.py 1.1 => 1.1.4.1 ===
product['dependent_products'] = tuple(map(str, dep_ids))
-# Functions for sorting modules by dependency.
-
-def listRequiredModulesByClass(klass, rval):
- if hasattr(klass, '__module__'):
- rval[klass.__module__] = 1 # klass.__module__ is a string.
- if hasattr(klass, '__bases__'):
- for b in klass.__bases__:
- listRequiredModulesByClass(b, rval)
-
-def listRequiredModules(module):
- rval = {}
-
- if hasattr(module, '__dict__'):
- for key, value in module.__dict__.items():
- t = type(value)
- if t in ClassTypes:
- listRequiredModulesByClass(value, rval)
- elif t is ModuleType and hasattr(value, '__name__'):
- rval[value.__name__] = 1
- elif t is FuncType and value.func_globals.has_key('__name__'):
- rval[value.func_globals['__name__']] = 1
- return rval
-
-def sortModulesByDependency(modlist):
- unchosen = {}
- for name, module in modlist:
- unchosen[name] = (module, listRequiredModules(module))
- chose = 1
- rval = []
- while chose:
- chose = 0
- for name, (module, req) in unchosen.items():
- all_satisfied = 1
- for n in unchosen.keys():
- if name == n:
- continue # Skip self.
- if req.has_key(n):
- # There is still a dependency. Can't
- # include this module in the list yet.
- all_satisfied = 0
- break
- if all_satisfied:
- chose = 1
- rval.append((name, module))
- del unchosen[name]
- # There might be some modules left over that are interdependent.
- for name, (module, req) in unchosen.items():
- rval.append((name, module))
- return rval
# Functions for performing refresh.
@@ -250,7 +201,7 @@
for name, module in sys.modules.items():
if module and (name == prefix or name[:lpdot] == prefixdot):
reload_var = getReloadVar(module)
- if callable(reload_var) or reload_var:
+ if reload_var:
rval.append((name, module))
return rval
@@ -276,33 +227,27 @@
setupModTimes(productid) # Refresh again only if changed again.
modlist = listRefreshableModules(productid)
- modlist = sortModulesByDependency(modlist)
-
- for name, module in modlist:
- # Remove the __import_error__ attribute.
- try: del module.__import_error__
- except: pass
- # Ask the module how it should be reloaded.
- reload_var = getReloadVar(module)
- if callable(reload_var):
- try:
- reload_var()
- except:
- logBadRefresh(productid)
- return 0
- else:
- try:
- reload(module)
- except:
- logBadRefresh(productid)
- return 0
-
- # Reinitialize and reinstall the product.
- from OFS import Application
- Application.reimport_product(productid)
- app = jar.root()['Application']
- Application.reinstall_product(app, productid)
- return 1
+ former_modules = {}
+ try:
+ # Remove modules from sys.modules but keep a handle
+ # on the old modules in case there's a problem.
+ for name, module in modlist:
+ m = sys.modules.get(name, None)
+ if m is not None:
+ former_modules[name] = m
+ del sys.modules[name]
+
+ # Reimport and reinstall the product.
+ from OFS import Application
+ Application.reimport_product(productid)
+ app = jar.root()['Application']
+ Application.reinstall_product(app, productid)
+ return 1
+ except:
+ # Couldn't refresh. Reinstate removed modules.
+ for name, module in former_modules.items():
+ sys.modules[name] = module
+ raise
def performSafeRefresh(jar, productid):
try:
@@ -316,6 +261,7 @@
return 1
def performFullRefresh(jar, productid):
+ # Refresh dependent products also.
if performSafeRefresh(jar, productid):
dep_ids = getDependentProducts(jar, productid)
for dep_id in dep_ids:
@@ -412,6 +358,9 @@
# Functions for performing auto-refresh.
def checkAutoRefresh(jar):
+ '''
+ Returns the IDs of products that need to be auto-refreshed.
+ '''
# Note: this function is NOT allowed to change the database!
global next_auto_refresh_check
now = time()