|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118 |
-
-
-
-
-
-
-
-
-
-
-
-
-
- """Class advice.
-
- This module was adapted from 'protocols.advice', part of the Python
- Enterprise Application Kit (PEAK). Please notify the PEAK authors
- (pje@telecommunity.com and tsarna@sarna.org) if bugs are found or
- Zope-specific changes are required, so that the PEAK version of this module
- can be kept in sync.
-
- PEAK is a Python application framework that interoperates with (but does
- not require) Zope 3 and Twisted. It provides tools for manipulating UML
- models, object-relational persistence, aspect-oriented programming, and more.
- Visit the PEAK home page at http://peak.telecommunity.com for more information.
- """
-
- from types import FunctionType
-
- __all__ = [
- 'determineMetaclass',
- 'getFrameInfo',
- 'isClassAdvisor',
- 'minimalBases',
- ]
-
- import sys
-
- def getFrameInfo(frame):
- """Return (kind,module,locals,globals) for a frame
-
- 'kind' is one of "exec", "module", "class", "function call", or "unknown".
- """
-
- f_locals = frame.f_locals
- f_globals = frame.f_globals
-
- sameNamespace = f_locals is f_globals
- hasModule = '__module__' in f_locals
- hasName = '__name__' in f_globals
-
- sameName = hasModule and hasName
- sameName = sameName and f_globals['__name__']==f_locals['__module__']
-
- module = hasName and sys.modules.get(f_globals['__name__']) or None
-
- namespaceIsModule = module and module.__dict__ is f_globals
-
- if not namespaceIsModule:
-
- kind = "exec"
- elif sameNamespace and not hasModule:
- kind = "module"
- elif sameName and not sameNamespace:
- kind = "class"
- elif not sameNamespace:
- kind = "function call"
- else:
-
-
- kind = "unknown"
- return kind, module, f_locals, f_globals
-
-
- def isClassAdvisor(ob):
- """True if 'ob' is a class advisor function"""
- return isinstance(ob,FunctionType) and hasattr(ob,'previousMetaclass')
-
-
- def determineMetaclass(bases, explicit_mc=None):
- """Determine metaclass from 1+ bases and optional explicit __metaclass__"""
-
- meta = [getattr(b,'__class__',type(b)) for b in bases]
-
- if explicit_mc is not None:
-
-
- meta.append(explicit_mc)
-
- if len(meta)==1:
-
- return meta[0]
-
- candidates = minimalBases(meta)
-
- if len(candidates)>1:
-
- raise TypeError("Incompatible metatypes", bases)
-
-
- return candidates[0]
-
-
- def minimalBases(classes):
- """Reduce a list of base classes to its ordered minimum equivalent"""
- candidates = []
-
- for m in classes:
- for n in classes:
- if issubclass(n,m) and m is not n:
- break
- else:
-
- if m in candidates:
- candidates.remove(m)
- candidates.append(m)
-
- return candidates
|