childNodePair
                );
    }
    /**
     * Get a collection of all child association references for a given parent node.
     * 
     * WARNING: Be sure selective when doing this call recursively.
     * 
     * @param parentNodeId          the parent node
     * @param resultsCallback       the callback that will be called with the results
     * @param recurse               if true then iterate over the entire tree of nodes.
     *                              Resursion is done top-down i.e. the first level children are all
     *                              enumerated first, followed by all second level children and so on.
     */
    @DirtySessionAnnotation(markDirty=false)
    public void getChildAssocs(Long parentNodeId, ChildAssocRefQueryCallback resultsCallback, boolean recurse);
    
    /**
     * Get a collection of all child association references for a given parent node.
     * 
     * @param parentNodeId the parent node
     * @param resultsCallback       the callback that will be called with the results
     */
    @DirtySessionAnnotation(markDirty=false)
    public void getChildAssocs(Long parentNodeId, QName assocQName, ChildAssocRefQueryCallback resultsCallback);
    
    @DirtySessionAnnotation(markDirty=false)
    public void getChildAssocsByTypeQNames(
            Long parentNodeId,
            List assocTypeQNames,
            ChildAssocRefQueryCallback resultsCallback);
    
    @DirtySessionAnnotation(markDirty=false)
    public void getChildAssocsByTypeQNameAndQName(
            Long parentNodeId,
            QName assocTypeQName,
            QName assocQName,
            ChildAssocRefQueryCallback resultsCallback);
    
    @DirtySessionAnnotation(markDirty=false)
    public void getPrimaryChildAssocs(Long parentNodeId, ChildAssocRefQueryCallback resultsCallback);
    
    @DirtySessionAnnotation(markDirty=false)
    public void getPrimaryChildAssocsNotInSameStore(Long parentNodeId, ChildAssocRefQueryCallback resultsCallback);
    
    /**
     * Interface used to iterate over pure node results
     * @author Derek Hulley
     */
    public interface NodeRefQueryCallback
    {
        /**
         * 
         * @param nodePair          the node result
         * @return                  Returns true if more results are required
         */
        boolean handle(Pair nodePair);
    }
    
    @DirtySessionAnnotation(markDirty=false)
    public void getNodesWithChildrenInDifferentStores(Long minNodeId, int count, NodeRefQueryCallback resultsCallback);
    
    @DirtySessionAnnotation(markDirty=false)
    public void getNodesWithAspect(QName aspectQName, Long minNodeId, int count, NodeRefQueryCallback resultsCallback);
    
    /**
     * @return Returns an association matching the given parent, type and child name - or null
     *      if not found
     */
    @DirtySessionAnnotation(markDirty=false)
    public Pair getChildAssoc(Long parentNodeId, QName assocTypeQName, String childName);
    
    /**
     * @return Returns a matching association or null if one was not found
     * 
     * @see ChildAssoc
     */
    @DirtySessionAnnotation(markDirty=false)
    public Pair getChildAssoc(
            Long parentNodeId,
            Long childNodeId,
            QName assocTypeQName,
            QName qname);
    /**
     * Deletes an explicit child association.
     * 
     * @return Returns true if the association was deleted, otherwise false
     */
    @DirtySessionAnnotation(markDirty=true)
    public boolean deleteChildAssoc(
            final Long parentNodeId,
            final Long childNodeId,
            final QName assocTypeQName,
            final QName qname);
    
    /**
     * @param assoc the child association to remove
     */
    @DirtySessionAnnotation(markDirty=true)
    public void deleteChildAssoc(Long childAssocId);
    
    /**
     * Finds the association between the node's primary parent and the node itself
     */
    @DirtySessionAnnotation(markDirty=false)
    public Pair getPrimaryParentAssoc(Long childNodeId);
    
    /**
     * Get all parent associations for the node.  This methods includes a cache safety check.
     * @param childNode the child node
     * @return Returns all parent associations for the node.
     */
    @DirtySessionAnnotation(markDirty=false)
    public Collection> getParentAssocs(final Long childNodeId);
    
    /**
     * @return Returns the persisted and filled association
     * @see NodeAssoc
     */
    @DirtySessionAnnotation(markDirty=true)
    public Pair newNodeAssoc(
            Long sourceNodeId,
            Long targetNodeId,
            QName assocTypeQName);
    
    /**
     * @return Returns a list of all node associations associated with the given node
     */
    @DirtySessionAnnotation(markDirty=false)
    public Collection> getNodeAssocsToAndFrom(final Long nodeId);
    /**
     * @return Returns the node association or null if not found
     */
    @DirtySessionAnnotation(markDirty=false)
    public Pair getNodeAssoc(Long sourceNodeId, Long targetNodeId, QName assocTypeQName);
    
    /**
     * @return Returns all the node associations where the node is the source
     */
    @DirtySessionAnnotation(markDirty=false)
    public Collection> getTargetNodeAssocs(Long sourceNodeId);
    
    /**
     * @return Returns all the node associations where the node is the target
     */
    @DirtySessionAnnotation(markDirty=false)
    public Collection> getSourceNodeAssocs(Long targetNodeId);
    
    /**
     * @param assoc the node association to remove
     */
    @DirtySessionAnnotation(markDirty=true)
    public void deleteNodeAssoc(Long assocId);
    
    /**
     * Iterate over all nodes that have a given property type with a given string value.
     * 
     * @param storeRef                          the store to search in
     * @param propertyQName                     the qualified name of the property
     * @param value                             the string value to match
     * @param handler                           the callback to use while iterating over the URLs
     * @return Returns the values for the given type definition
     */
    @DirtySessionAnnotation(markDirty=true)
    public void getPropertyValuesByPropertyAndValue(
            StoreRef storeRef,
            QName propertyQName,
            String value,
            NodePropertyHandler handler);
    
    /**
     * Iterate over all property values for the given type definition.  This will also dig out values that
     * were persisted as type d:any.
     * 
     * @param actualDataTypeDefinition          the persisted type to retrieve
     * @param handler                           the callback to use while iterating over the URLs
     * @return Returns the values for the given type definition
     */
    @DirtySessionAnnotation(markDirty=true)
    public void getPropertyValuesByActualType(DataTypeDefinition actualDataTypeDefinition, NodePropertyHandler handler);
    
    /**
     * @return      Returns the total number of nodes in the ADM repository
     */
    @DirtySessionAnnotation(markDirty=false)
    public int getNodeCount();
    /**
     * @return      Returns the total number of nodes in the ADM store
     */
    @DirtySessionAnnotation(markDirty=false)
    public int getNodeCount(final StoreRef storeRef);
    
    /**
     * Iterface to handle callbacks when iterating over properties
     * 
     * @author Derek Hulley
     * @since 2.0
     */
    public interface NodePropertyHandler
    {
        void handle(NodeRef nodeRef, QName nodeTypeQName, QName propertyQName, Serializable value);
    }
    
    @DirtySessionAnnotation(markDirty=true)
    public Transaction getTxnById(long txnId);
    /**
     * Get all transactions in a given time range.  Since time-based retrieval doesn't guarantee uniqueness
     * for any given millisecond, a list of optional exclusions may be provided.
     * 
     * @param excludeTxnIds         a list of txn IDs to ignore.  null is allowed.
     * @param remoteOnly            true if locally-written transactions must be ignored
     */
    @DirtySessionAnnotation(markDirty=true)
    public List getTxnsByCommitTimeAscending(
            long fromTimeInclusive,
            long toTimeExclusive,
            int count,
            List excludeTxnIds,
            boolean remoteOnly);
    /**
     * Get all transactions in a given time range.  Since time-based retrieval doesn't guarantee uniqueness
     * for any given millisecond, a list of optional exclusions may be provided.
     * 
     * @param excludeTxnIds         a list of txn IDs to ignore.  null is allowed.
     * @param remoteOnly            true if locally-written transactions must be ignored
     */
    @DirtySessionAnnotation(markDirty=true)
    public List getTxnsByCommitTimeDescending(
            long fromTimeInclusive,
            long toTimeExclusive,
            int count,
            List excludeTxnIds,
            boolean remoteOnly);
    /**
     * Get the lowest commit time for a set of transactions
     * 
     * @param includeTxnIds     a list of transaction IDs to search for
     * @return      Returns the transactions by commit time for the given IDs
     */
    @DirtySessionAnnotation(markDirty=true)
    public List getTxnsByMinCommitTime(List includeTxnIds);
    
    @DirtySessionAnnotation(markDirty=false)
    public int getTxnUpdateCount(final long txnId);
    @DirtySessionAnnotation(markDirty=false)
    public int getTxnDeleteCount(final long txnId);
    
    @DirtySessionAnnotation(markDirty=false)
    public int getTransactionCount();
    
    @DirtySessionAnnotation(markDirty=false)
    public List getTxnChangesForStore(final StoreRef storeRef, final long txnId);
    
    @DirtySessionAnnotation(markDirty=false)
    public List getTxnChanges(final long txnId);
}