ó
H`¾Tc           @   sâ   d  Z  d d l m Z m Z m Z m Z d d l m Z d d l m Z d e f d „  ƒ  YZ	 d e
 f d	 „  ƒ  YZ d
 e
 f d „  ƒ  YZ e ƒ  Z e j Z Z e j Z Z e j Z Z d „  Z d „  Z d „  Z d „  Z d S(   sœ  Defines SQLAlchemy's system of class instrumentation.

This module is usually not directly visible to user applications, but
defines a large part of the ORM's interactivity.

instrumentation.py deals with registration of end-user classes
for state tracking.   It interacts closely with state.py
and attributes.py which establish per-instance and per-class-attribute
instrumentation, respectively.

The class instrumentation system can be customized on a per-class
or global basis using the :mod:`sqlalchemy.ext.instrumentation`
module, which provides the means to build and specify
alternate instrumentation forms.

.. versionchanged: 0.8
   The instrumentation extension system was moved out of the
   ORM and into the external :mod:`sqlalchemy.ext.instrumentation`
   package.  When that package is imported, it installs
   itself within sqlalchemy.orm so that its more comprehensive
   resolution mechanics take effect.

i   (   t   exct   collectionst
   interfacest   statei   (   t   util(   t   baset   ClassManagerc           B   sè  e  Z d  Z e j Z e j Z d( Z	 e
 j Z d( Z d „  Z d „  Z d „  Z e d „  ƒ Z e j d „  ƒ Z d( d „ Z d „  Z d „  Z d	 „  Z d
 „  Z e j d „  ƒ Z d „  Z d „  Z e j d „  ƒ Z e j d „  ƒ Z e j d „  ƒ Z  e! d „ Z" d „  Z# d „  Z$ e! d „ Z% d „  Z& d „  Z' d „  Z( d „  Z) d „  Z* d „  Z+ d „  Z, e! d „ Z- d „  Z. e d „  ƒ Z/ d( d „ Z0 d( d  „ Z1 d! „  Z2 d" „  Z3 d# „  Z4 d$ „  Z5 e! d% „ Z6 d& „  Z7 e7 Z8 d' „  Z9 RS()   s,   tracks state information at the class level.c         C   s7  | |  _  i  |  _ d  |  _ i  |  _ i  |  _ g  g  |  j  j D]! } t | t ƒ r= t	 | ƒ ^ q= D] } | d  k	 rb | ^ qb |  _
 x |  j
 D] } |  j | ƒ q W|  j j j | |  ƒ x? | j D]4 } t	 | ƒ } | d  k	 rÄ |  j j | j ƒ qÄ qÄ W|  j ƒ  |  j ƒ  d | j k r3t j d | ƒ n  d  S(   Nt   __del__s¦   __del__() method on class %s will cause unreachable cycles and memory leaks, as SQLAlchemy instrumentation often creates reference cycles.  Please remove this method.(   t   class_t   infot   Nonet   new_initt   local_attrst	   originalst	   __bases__t
   isinstancet   typet   manager_of_classt   _basest   updatet   dispatcht   _eventst   _new_classmanager_instancet   __mro__t   _updatet   managet   _instrument_initt   __dict__R   t   warn(   t   selfR   R   t   mgrt   basecls(    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   __init__3   s,    					%

	c         C   s
   t  |  ƒ S(   N(   t   id(   R   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   __hash__T   s    c         C   s
   | |  k S(   N(    (   R   t   other(    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   __eq__W   s    c         C   s   d |  j  k S(   Nt   mapper(   R   (   R   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt	   is_mappedZ   s    c         C   s   t  j |  j ƒ ‚ d  S(   N(   R    t   UnmappedClassErrorR   (   R   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyR%   ^   s    c         c   s   | d k r t ƒ  } n  xq |  j j D]c } xZ t | j ƒ j | ƒ D]@ } | j | ƒ | j | } t | t j	 ƒ rD | | f VqD qD Wq% Wd S(   s  return an iterator of all classbound attributes that are
        implement :class:`._InspectionAttr`.

        This includes :class:`.QueryableAttribute` as well as extension
        types such as :class:`.hybrid_property` and
        :class:`.AssociationProxy`.

        N(
   R
   t   setR   R   R   t
   differencet   addR   R   t   _InspectionAttr(   R   t   excludet   superclst   keyt   val(    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   _all_sqla_attributesc   s    	c         C   s   | |  k o |  | j  d k	 S(   s\   Return True if the given attribute is fully initialized.

        i.e. has an impl.
        N(   t   implR
   (   R   R.   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   _attr_has_implu   s    c         C   s.   t  | ƒ } | d k r* t j | ƒ } n  | S(   sÏ  Create a new ClassManager for a subclass of this ClassManager's
        class.

        This is called automatically when attributes are instrumented so that
        the attributes can be propagated to subclasses against their own
        class-local manager, without the need for mappers etc. to have already
        pre-configured managers for the full class hierarchy.   Mappers
        can post-configure the auto-generated ClassManager when needed.

        N(   R   R
   t   _instrumentation_factoryt   create_manager_for_cls(   R   t   clst   manager(    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   _subclass_manager}   s    c         C   s;   |  j  j |  _ t |  j  |  ƒ |  _ |  j d |  j ƒ d  S(   NR    (   R   R    t   original_initt   _generate_initR   t   install_member(   R   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyR      s    c         C   s&   |  j  r" |  j d ƒ d  |  _  n  d  S(   NR    (   R   t   uninstall_memberR
   (   R   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   _uninstrument_init—   s    	c         C   s   |  j  j |  |  j ƒ t j S(   N(   R   t
   first_initR   R   t   InstanceState(   R   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   _state_constructorœ   s    c         C   s   t  |  j |  j |  ƒ d S(   s0   Mark this instance as the manager for its class.N(   t   setattrR   t   MANAGER_ATTR(   R   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyR   ¡   s    c         C   s   t  |  j |  j ƒ d S(   s)   Dissasociate this manager from its class.N(   t   delattrR   RA   (   R   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   dispose¦   s    c         C   s   t  S(   N(   t   _default_manager_getter(   R   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   manager_getter«   s    c         C   s   t  S(   sÍ   Return a (instance) -> InstanceState callable.

        "state getter" callables should raise either KeyError or
        AttributeError if no InstanceState could be found for the
        instance.
        (   t   _default_state_getter(   R   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   state_getter¯   s    	c         C   s   t  S(   N(   t   _default_dict_getter(   R   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   dict_getterº   s    c         C   sƒ   | r | |  j  k r9 d  Sn | |  j  | <|  j | | ƒ | |  | <x9 |  j j ƒ  D]( } |  j | ƒ } | j | | t ƒ qS Wd  S(   N(   R   t   install_descriptorR   t   __subclasses__R7   t   instrument_attributet   True(   R   R.   t   instt
   propagatedR5   R6   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyRL   ¾   s    
c         c   sr   xk |  j  j ƒ  D]Z } t | ƒ } | d  k	 r | |  k	 r | V| rj x | j t ƒ D] } | VqU Wqj q q Wd  S(   N(   R   RK   R   R
   t   subclass_managersRM   (   R   t	   recursiveR5   R   t   m(    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyRP   Ë   s    c         C   s!   t  j j |  j | |  | ƒ d  S(   N(   R3   R   t   attribute_instrumentR   (   R   R.   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   post_configure_attributeÔ   s    	c         C   s   | |  k r d  S| r, | |  j  k rC d  Sn |  j  | =|  j | ƒ |  | =x< |  j j ƒ  D]+ } t | ƒ } | rZ | j | t ƒ qZ qZ Wd  S(   N(   R   t   uninstall_descriptorR   RK   R   t   uninstrument_attributeRM   (   R   R.   RO   R5   R6   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyRV   Ø   s    
c         C   sa   |  j  ƒ  d |  _ |  _ |  j j ƒ  x3 t |  ƒ D]% } | |  j k r4 |  j | ƒ q4 q4 Wd S(   s<   remove all instrumentation established by this ClassManager.N(	   R<   R
   R%   R   R	   t   cleart   listR   RV   (   R   R.   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt
   unregisterç   s    
c         C   sB   | |  j  |  j f k r+ t d | ƒ ‚ n  t |  j | | ƒ d  S(   NsW   %r: requested attribute name conflicts with instrumentation attribute of the same name.(   t
   STATE_ATTRRA   t   KeyErrorR@   R   (   R   R.   RN   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyRJ   ó   s    c         C   s   t  |  j | ƒ d  S(   N(   RB   R   (   R   R.   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyRU   ú   s    c         C   sd   | |  j  |  j f k r+ t d | ƒ ‚ n  |  j j | t |  j | d  ƒ ƒ t |  j | | ƒ d  S(   NsW   %r: requested attribute name conflicts with instrumentation attribute of the same name.(	   RZ   RA   R[   R   t
   setdefaultt   getattrR   R
   R@   (   R   R.   t   implementation(    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyR:   ý   s
    "c         C   s;   |  j  j | d  ƒ } | d  k	 r7 t |  j | | ƒ n  d  S(   N(   R   t   popR
   R@   R   (   R   R.   t   original(    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyR;     s    c         C   s   t  j | ƒ S(   N(   R   t   prepare_instrumentation(   R   R.   t   collection_class(    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   instrument_collection_class
  s    c         C   s1   | ƒ  } t  j |  j | ƒ | | ƒ } | | f S(   N(   R   t   CollectionAdaptert   get_impl(   R   R.   R   t   factoryt	   user_datat   adapter(    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   initialize_collection  s    	c         C   s!   | r | |  k S| |  j  k Sd  S(   N(   R   (   R   R.   t   search(    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   is_instrumented  s    
c         C   s   |  | j  S(   N(   R1   (   R   R.   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyRe     s    c         C   s   t  |  j ƒ  ƒ S(   N(   t   itert   values(   R   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt
   attributes  s    c         C   s>   |  j  j |  j  ƒ } t | |  j | p6 |  j | |  ƒ ƒ | S(   N(   R   t   __new__R@   RZ   R?   (   R   R   t   instance(    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   new_instance"  s    c         C   s)   t  | |  j | p! |  j | |  ƒ ƒ d  S(   N(   R@   RZ   R?   (   R   Rp   R   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   setup_instance(  s    c         C   s   t  | |  j ƒ d  S(   N(   RB   RZ   (   R   Rp   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   teardown_instance,  s    c         C   s   t  | | ƒ S(   N(   t   _SerializeManager(   R   R   t
   state_dict(    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt
   _serialize/  s    c         C   sw   t  | |  j ƒ r t S|  j | j k	 rJ |  j rJ |  j | j ƒ j | ƒ S|  j | |  ƒ } t	 | |  j | ƒ | Sd S(   sƒ   Install a default InstanceState if none is present.

        A private convenience method used by the __init__ decorator.

        N(
   t   hasattrRZ   t   FalseR   t	   __class__R&   R7   t   _new_state_if_noneR?   R@   (   R   Rp   R   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyRz   2  s    	c         C   s   t  | |  j ƒ S(   N(   Rw   RZ   (   R   Rp   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt	   has_stateH  s    c         C   s   |  j  | ƒ j | d | ƒS(   t   TODOt
   optimistic(   Re   t	   hasparent(   R   R   R.   R}   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt
   has_parentK  s    c         C   s   t  S(   s=   All ClassManagers are non-zero regardless of attribute state.(   RM   (   R   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   __bool__O  s    c         C   s    d |  j  j |  j t |  ƒ f S(   Ns   <%s of %r at %x>(   Ry   t   __name__R   R!   (   R   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   __repr__U  s    N(:   R   t
   __module__t   __doc__R   t   DEFAULT_MANAGER_ATTRRA   t   DEFAULT_STATE_ATTRRZ   R
   t   deferred_scalar_loadert   objectR    R8   Rf   R"   R$   t   propertyR&   R   t   memoized_propertyR%   R0   R2   R7   R   R<   R?   R   RC   t   hybridmethodRE   RG   RI   Rx   RL   RP   RT   RV   RY   RJ   RU   R:   R;   Rc   Ri   Rk   Re   Rn   Rq   Rr   Rs   Rv   Rz   R{   R   R€   t   __nonzero__R‚   (    (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyR   &   s\   				!					
																			Rt   c           B   s    e  Z d  Z d „  Z d „  Z RS(   s¡   Provide serialization of a :class:`.ClassManager`.

    The :class:`.InstanceState` uses ``__init__()`` on serialize
    and ``__call__()`` on deserialize.

    c         C   s,   | j  |  _  | j } | j j | | ƒ d  S(   N(   R   R6   R   t   pickle(   R   R   t   dR6   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyR    c  s    	c         C   sš   t  |  j ƒ | _ } | d  k r> t j | d |  j ƒ ‚ n& | j rd | j j rd | j j	 ƒ  n  | d  k	 rƒ | j
 | | ƒ n  | j j | | ƒ d  S(   Nsx   Cannot deserialize object of type %r - no mapper() has been configured for this class within the current Python process!(   R   R   R6   R
   R    t   UnmappedInstanceErrorR&   R%   t
   configuredt   _configure_allRr   R   t   unpickle(   R   R   RN   Ru   R6   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   __call__h  s    (   R   Rƒ   R„   R    R“   (    (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyRt   Z  s   	t   InstrumentationFactoryc           B   s2   e  Z d  Z d „  Z d „  Z d „  Z d „  Z RS(   s'   Factory for new ClassManager instances.c         C   s   | d  k	 s t ‚ t | ƒ d  k s* t ‚ |  j | ƒ \ } } | d  k r` t } | | ƒ } n  |  j | | ƒ | | _ |  j j | ƒ | S(   N(	   R
   t   AssertionErrorR   t   _locate_extended_factoryR   t   _check_conflictsRf   R   t   class_instrument(   R   R   R6   Rf   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyR4     s    	c         C   s   d S(   s2   Overridden by a subclass to do an extended lookup.N(   NN(   R
   (   R   R   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyR–   ”  s    c         C   s   d S(   s;   Overridden by a subclass to test for conflicting factories.N(    (   R   R   Rf   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyR—   ˜  s    c         C   sY   t  | ƒ } | j ƒ  | j ƒ  |  j j | ƒ t j | j k rU t | t j ƒ n  d  S(   N(	   R   RY   RC   R   t   class_uninstrumentR   RA   R   RB   (   R   R   R6   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyRY   œ  s    

(   R   Rƒ   R„   R4   R–   R—   RY   (    (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyR”   }  s
   			c         C   s.   t  |  ƒ } | d k r* t j |  ƒ } n  | S(   s_   Register class instrumentation.

    Returns the existing or newly created class manager.

    N(   R   R
   R3   R4   (   R   R6   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   register_class²  s    c         C   s   t  j |  ƒ d S(   s!   Unregister class instrumentation.N(   R3   RY   (   R   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   unregister_class¿  s    c         C   s   t  |  j ƒ j | d t ƒS(   sï   Return True if the given attribute on the given instance is
    instrumented by the attributes package.

    This function may be used regardless of instrumentation
    applied directly to the class, i.e. no descriptors are required.

    Rj   (   R   Ry   Rk   RM   (   Rp   R.   (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyRk   Å  s    c         B   só   |  j  } | s t ‚ d } e j | d e ƒ} | | } e j rj e | d | ƒ } e | d d ƒ } n$ e | d d ƒ } e | d d ƒ } e ƒ  j	 ƒ  }	 | |	 f d U|	 d }
 | j
 |
 _
 | rÓ | |
 _ n  e j rï | rï | |
 _ n  |
 S(	   s>   Build an __init__ decorator that triggers ClassManager events.sâ   def __init__(%(apply_pos)s):
    new_state = class_manager._new_state_if_none(%(self_arg)s)
    if new_state:
        return new_state._initialize_instance(%(apply_kw)s)
    else:
        return original__init__(%(apply_kw)s)
t   groupedt   im_funct   func_defaultst   __defaults__t   __kwdefaults__NR    (   R    R•   R   t   format_argspec_initRx   t   py2kR]   R
   t   localst   copyR„   RŸ   R    (   R   t   class_managert   original__init__t	   func_bodyt	   func_varst	   func_textt   funcRž   t   func_kw_defaultst   envR    (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyR9   Ñ  s&    	
	
N(   R„   t    R    R   R   R   R   R   t   dictR   Rˆ   Rt   R”   R3   t   instance_stateRF   t   instance_dictRH   R   RD   Rš   R›   Rk   R9   (    (    (    sd   /var/www/send.findwatt.com/datamanager/lib/python2.7/site-packages/sqlalchemy/orm/instrumentation.pyt   <module>   s   "ÿ 5#)				