634 lines
25 KiB
Java

/*
* Copyright (C) 2005-2010 Alfresco Software Limited.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
* As a special exception to the terms and conditions of version 2.0 of
* the GPL, you may redistribute this Program in connection with Free/Libre
* and Open Source Software ("FLOSS") applications as described in Alfresco's
* FLOSS exception. You should have recieved a copy of the text describing
* the FLOSS exception, and it is also available here:
* http://www.alfresco.com/legal/licensing"
*/
package org.alfresco.repo.domain.node;
import java.io.Serializable;
import java.util.Collection;
import java.util.List;
import java.util.Map;
import java.util.Set;
import org.alfresco.repo.node.NodeBulkLoader;
import org.alfresco.service.cmr.dictionary.InvalidTypeException;
import org.alfresco.service.cmr.repository.AssociationRef;
import org.alfresco.service.cmr.repository.ChildAssociationRef;
import org.alfresco.service.cmr.repository.InvalidNodeRefException;
import org.alfresco.service.cmr.repository.NodeRef;
import org.alfresco.service.cmr.repository.Path;
import org.alfresco.service.cmr.repository.StoreExistsException;
import org.alfresco.service.cmr.repository.StoreRef;
import org.alfresco.service.namespace.QName;
import org.alfresco.util.Pair;
/**
* DAO services for <b>alf_node</b> and related tables
*
* @author Derek Hulley
* @since 3.4
*/
public interface NodeDAO extends NodeBulkLoader
{
/**
* Interface used to iterate over pure node results
*
* @author Derek Hulley
* @since 3.4
*/
public interface NodeRefQueryCallback
{
/**
* @param nodePair the node result
* @return Returns <tt>true</tt> if more results are required
*/
boolean handle(Pair<Long, NodeRef> nodePair);
}
/*
* Transaction
*/
/**
*
* @return Returns the ID of the current transaction entry
*/
public Long getCurrentTransactionId();
/*
* Store
*/
/**
* Fetch a list of all stores in the repository
*
* @return Returns a list of stores
*/
public List<Pair<Long, StoreRef>> getStores();
/**
* Find out if a store exists or not
*
* @param storeRef the store
* @return Returns <tt>true</tt> if the store exists otherwise <tt>false</tt>
*/
public boolean exists(StoreRef storeRef);
/**
* Creates a unique store for the given protocol and identifier combination.
* The root node is created with the "root" aspect.
*
* @return Returns the root node, which is added automatically.
* @throws StoreExistsException if the store already exists
*/
public Pair<Long, NodeRef> newStore(StoreRef storeRef);
public Pair<Long, NodeRef> getRootNode(StoreRef storeRef);
/*
* Node
*/
/**
* Find out if a node exists. Unpurged deleted nodes do not count as they are the DAO's concern only.
*
* @param nodeRef the potentially valid node reference
* @return Returns <tt>true</tt> if the node is present and undeleted
*/
public boolean exists(NodeRef nodeRef);
/**
* Get the current status of the node, including deleted nodes.
*
* @param nodeRef the node reference
* @return Returns the current status of the reference.
* This will only be <tt>null</tt> if the node never existed or has been
* purged following deletion.
*/
public NodeRef.Status getNodeRefStatus(NodeRef nodeRef);
public Pair<Long, NodeRef> getNodePair(NodeRef nodeRef);
public Pair<Long, NodeRef> getNodePair(Long nodeId);
public QName getNodeType(Long nodeId);
public Long getNodeAclId(Long nodeId);
/**
* Create a new node. Note that allowing the <b>uuid</b> to be assigned by passing in a <tt>null</tt>
* is more efficient.
*
* @param parentNodeId the ID of the parent node (may not be <tt>null</tt>)
* @param assocTypeQName the primary association (may not be <tt>null</tt>)
* @param assocQName the association path (may not be <tt>null</tt>)
* @param storeRef the store to which the node must belong
* @param uuid the node store-unique identifier, or <tt>null</tt> to assign a GUID
* @param nodeTypeQName the type of the node
* @param childNodeName the <b>cm:name</b> of the child node or <tt>null</tt> to use the node's UUID
* @param auditableProperties a map containing any <b>cm:auditable</b> properties for the node
* @return Returns the details of the child association created
* @throws InvalidTypeException if the node type is invalid or if the node type
* is not a valid real node
*/
public ChildAssocEntity newNode(
Long parentNodeId,
QName assocTypeQName,
QName assocQName,
StoreRef storeRef,
String uuid,
QName nodeTypeQName,
String childNodeName,
Map<QName, Serializable> auditableProperties/*,
Map<QName, Serializable> ownableProperties*/) throws InvalidTypeException;
/**
* Update a node's primary association, giving it a new parent and new association parameters.
* <p/>
* <b>**NEW**:</b> If the parent node's store differs from the child node's store, then the
* child node's store is updated. Store move conflicts are automatically handled by assigning
* new UUIDs to the existing target node.
*
* @param childNodeId the child node that is moving
* @param newParentNodeId the new parent node (may not be <tt>null</tt>)
* @param assocTypeQName the new association type or <tt>null</tt> to keep the existing type
* @param assocQName the new association qname or <tt>null</tt> to keep the existing name
* @return Returns the new association reference
*/
public Pair<Long, ChildAssociationRef> moveNode(
Long childNodeId,
Long newParentNodeId,
QName assocTypeQName,
QName assocQName);
/**
* @param storeRef the new store or <tt>null</tt> to keep the existing one
* @param uuid the new UUID for the node or <tt>null</tt> to keep it the same
* @param nodeTypeQName the new type QName for the node or <tt>null</tt> to keep the existing one
*/
public void updateNode(Long nodeId, StoreRef storeRef, String uuid, QName nodeTypeQName);
public void setNodeAclId(Long nodeId, Long aclId);
public void setPrimaryChildrenSharedAclId(
Long primaryParentNodeId,
Long optionalOldSharedAlcIdInAdditionToNull,
Long newSharedAclId);
/**
* Deletes the node and all entities. Note that the node entry will still exist and be
* associated with a live transaction.
*/
public void deleteNode(Long nodeId);
/**
* Remove all traces of the node. This assumes that the node has been marked
* for deletion using {@link #deleteNode(Long)}.
*
* @deprecated This will be replaced with a purgeNodes(long maxTxnCommitTimeMs)
*/
public void purgeNode(Long nodeId);
/*
* Properties
*/
public Serializable getNodeProperty(Long nodeId, QName propertyQName);
public Map<QName, Serializable> getNodeProperties(Long nodeId);
public boolean setNodeProperties(Long nodeId, Map<QName, Serializable> properties);
public boolean addNodeProperty(Long nodeId, QName qname, Serializable value);
public boolean addNodeProperties(Long nodeId, Map<QName, Serializable> properties);
public boolean removeNodeProperties(Long nodeId, Set<QName> propertyQNames);
/*
* Aspects
*/
public Set<QName> getNodeAspects(Long nodeId);
public boolean hasNodeAspect(Long nodeId, QName aspectQName);
public boolean addNodeAspects(Long nodeId, Set<QName> aspectQNames);
public boolean removeNodeAspects(Long nodeId);
public boolean removeNodeAspects(Long nodeId, Set<QName> aspectQNames);
public void getNodesWithAspect(QName aspectQName, Long minNodeId, int count, NodeRefQueryCallback resultsCallback);
/*
* Node Assocs
*/
public Long newNodeAssoc(
Long sourceNodeId,
Long targetNodeId,
QName assocTypeQName);
/**
* Remove a specific node association
*
* @param assocId the node association ID to remove
* @return Returns the number of associations removed
*/
public int removeNodeAssoc(Long sourceNodeId, Long targetNodeId, QName assocTypeQName);
/**
* Remove all node associations that share the given node.
*
* @param nodeId the source or target of the associations
* @return Returns the number of associations removed
*/
public int removeNodeAssocsToAndFrom(Long nodeId);
/**
* Remove all node associations of given types that share the given node.
*
* @param nodeId the source or target of the associations
* @param assocTypeQNames the types that should be deleted
* @return Returns the number of associations removed
*/
public int removeNodeAssocsToAndFrom(Long nodeId, Set<QName> assocTypeQNames);
/**
* @return Returns all the node associations where the node is the </b>target</b>
*/
public Collection<Pair<Long, AssociationRef>> getSourceNodeAssocs(Long targetNodeId);
/**
* @return Returns all the node associations where the node is the <b>source</b>
*/
public Collection<Pair<Long, AssociationRef>> getTargetNodeAssocs(Long sourceNodeId);
/**
* @return Returns a specific node association with the given ID
*
* @throws ConcurrencyFailureException if the association ID is invalid
*/
public Pair<Long, AssociationRef> getNodeAssoc(Long assocId);
/*
* Child Assocs
*/
/**
* Interface used to iterate over results from child association queries
*
* @author Derek Hulley
* @since 3.4
*/
public interface ChildAssocRefQueryCallback
{
/**
* @return Return <tt>false</tt> to terminate the query
* i.e. stop receiving results
*/
boolean handle(
Pair<Long, ChildAssociationRef> childAssocPair,
Pair<Long, NodeRef> parentNodePair,
Pair<Long, NodeRef> childNodePair
);
/**
* @return Return <tt>true</tt> if caching of the results is required
*/
boolean preLoadNodes();
/**
* Called once the iteration of results has concluded
*/
void done();
}
/**
* Create a new child association. The unique enforcement for <b>cm:name</b> will be done
* as part of the association creation i.e. there is no need to update it after the fact.
*
* @param childNodeName the <b>cm:name</b> to apply to the association
* @return Returns the persisted and filled association's ID
*/
public Pair<Long, ChildAssociationRef> newChildAssoc(
Long parentNodeId,
Long childNodeId,
QName assocTypeQName,
QName assocQName,
String childNodeName);
/**
* @param assocId the ID of the child association to delete
*/
public void deleteChildAssoc(Long assocId);
/**
* Sets the association index ordering.
*
* @param parentNodeId the parent node ID
* @param childNodeId the child node ID
* @param assocTypeQName the association type
* @param assocQName the association path qualified name
* @param newIndex the new index
* @return Returns the number of associations modified
*/
public int setChildAssocIndex(
Long parentNodeId,
Long childNodeId,
QName assocTypeQName,
QName assocQName,
int index);
/**
* Bulk-update all unique name (<b>cm:name</b>) index for parent associations of a given node.
*
* @param childNodeId the child node who's name is changing
* @param childName the new <b>cm:name</b> value
*/
public void setChildAssocsUniqueName(Long childNodeId, String childName);
/**
* Get a specific association
*
* @param assocId the ID of the association
* @return Returns the association reference or <tt>null</tt> if it doesn't exist
*/
public Pair<Long, ChildAssociationRef> getChildAssoc(Long assocId);
/**
* Get a specific child association given all the determining data.
* <p>
* The implementation may find multiple entries (there is no constraint to prevent it)
* although the <b>cm:name</b> constraint will normally prevent the association from
* being created twice. The lowest ID association will always be returned and the
* others will be cleaned up if the transaction is read-write.
*
* @return Returns a matching association or null if one was not found.
*/
public Pair<Long, ChildAssociationRef> getChildAssoc(
Long parentNodeId,
Long childNodeId,
QName assocTypeQName,
QName assocQName);
/**
* Get the child associations of a given parent node, optionally filtering on association <tt>QName</tt>
* and association type <tt>QName</tt>.
* <p/>
* This is an efficient query for node paths.
*
* @param parentNodeId the parent node ID
* @param childNodeId the child node ID to filter on; <tt>null</tt> for no filtering
* @param assocTypeQName the association type qname to filter on; <tt>null<tt> for no filtering
* @param assocQName the association qname to filter on; <tt>null</tt> for no filtering
* @param isPrimary filter for primary (<tt>true</tt>) or secondary associations;
* <tt>null</tt> for no filtering.
* @param sameStore <tt>null</tt> to ignore, <tt>true</tt> to only get children that are in the
* same store as the parent, or <tt>false</tt> to only get children that are in
* a different store from the parent.
* @param resultsCallback the callback that will be called with the results
*/
public void getChildAssocs(
Long parentNodeId,
Long childNodeId,
QName assocTypeQName,
QName assocQName,
Boolean isPrimary,
Boolean sameStore,
ChildAssocRefQueryCallback resultsCallback);
/**
* Get the child associations of a given parent node, optionally filtering on type <tt>QName</tt>.
*
* @param parentNodeId the parent node ID
* @param assocTypeQNames the association type qnames to filter on; <tt>null<tt> for no filtering
* @param resultsCallback the callback that will be called with the results
*/
public void getChildAssocs(
Long parentNodeId,
Set<QName> assocTypeQNames,
ChildAssocRefQueryCallback resultsCallback);
/**
* Get a child association for given parent node, association type and child node name (<b>cm:name</b>).
*
* @param parentNodeId the parent Node ID
* @param assocTypeQName the association type to filter on
* @param childName the <b>cm:name</b> value to filter on
* @return Returns an association matching the given parent, type and child name
* (<b>cm:name</b>) - or <tt>null</tt> if not found
*/
public Pair<Long, ChildAssociationRef> getChildAssoc(Long parentNodeId, QName assocTypeQName, String childName);
/**
* Get the child associations of a given parent node, filtering on type <tt>QName</tt> and
* the <b>cm:name</b> of the child nodes.
* <p>
* <b>NOTE: </b>This method only works if the association type fundamentally supports unique-name enforcement.
*
* @param parentNodeId the parent node
* @param assocTypeQName the type of the association to check; or <tt>null</tt> for no filtering.
* If the association type is not specified, then the same child node may be
* included several times.
* @param childNames the names of the child nodes (<b>cm:name</b>). These will be matched exactly.
* @param resultsCallback the callback that will be called with the results
*/
public void getChildAssocs(
Long parentNodeId,
QName assocTypeQName,
Collection<String> childNames,
ChildAssocRefQueryCallback resultsCallback);
public void getChildAssocsByChildTypes(
Long parentNodeId,
Set<QName> childNodeTypeQNames,
ChildAssocRefQueryCallback resultsCallback);
/**
* Gets the set of child associations of a certain parent node without parent associations of a certain type to
* other nodes with the same parent! In effect the 'orphans' with respect to a certain association type.
*
* @param parentNodeId the parent node ID
* @param assocTypeQName the association type QName
* @param resultsCallback the callback that will be called with the results
*/
public void getChildAssocsWithoutParentAssocsOfType(
final Long parentNodeId,
final QName assocTypeQName,
ChildAssocRefQueryCallback resultsCallback);
/**
* Finds the association between the node's primary parent and the node itself
*
* @return Returns the primary (defining) association or <tt>null</tt>
* if it is a root node
*/
public Pair<Long, ChildAssociationRef> getPrimaryParentAssoc(Long childNodeId);
/**
* Get the parent association of a given parent node, optionally filtering on association <tt>QName</tt>
* and association type <tt>QName</tt>.
* <p/>
* This is an efficient query for node paths.
*
* @param childNodeId the child node ID
* @param assocTypeQName the association type qname to filter on; <tt>null<tt> for no filtering
* @param assocQName the association qname to filter on; <tt>null</tt> for no filtering
* @param isPrimary filter for primary (<tt>true</tt>) or secondary associations;
* <tt>null</tt> for no filtering.
* @param resultsCallback the callback that will be called with the results
*/
public void getParentAssocs(
Long childNodeId,
QName assocTypeQName,
QName assocQName,
Boolean isPrimary,
ChildAssocRefQueryCallback resultsCallback);
/**
* Fetch all <i>primary</i> child node IDs and corresponding ACL IDs.
*
* @param nodeId the parent node ID
* @return Returns a list of Node ID - ACL ID pairs
*/
public List<NodeIdAndAclId> getPrimaryChildrenAcls(Long nodeId);
/**
* Build the paths for a node.
*
* When searching for <code>primaryOnly == true</code>, checks that there is exactly
* one path.
*
* @param currentNodePair the leave or child node to start with
* @param primaryOnly <tt>true</tt> to follow only primary parent associations
*/
public List<Path> getPaths(Pair<Long, NodeRef> nodePair, boolean primaryOnly) throws InvalidNodeRefException;
/*
* Transactions
*/
/**
* Gets a batch of deleted nodes in old transactions.
*
* @param minNodeId the minimum node ID
* @param maxCommitTime the maximum commit time (to set a minimum transaction age)
* @param count the maximum number of results (for batching)
* @param resultsCallback the callback to pass results back
*
* @deprecated {@link #purgeNode(Long)}
*/
public void getNodesDeletedInOldTxns(
Long minNodeId,
long maxCommitTime,
int count,
NodeRefQueryCallback resultsCallback);
/**
* Retrieves the maximum transaction ID for which the commit time is less than the given time.
*
* @param maxCommitTime the max commit time (ms)
* @return the last transaction <i>on or before</i> the given time
*/
public Long getMaxTxnIdByCommitTime(long maxCommitTime);
/**
* Retrieves a specific transaction.
*
* @param txnId the unique transaction ID.
* @return the requested transaction or <tt>null</tt>
*/
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. <tt>null</tt> is allowed.
* @param remoteOnly <tt>true</tt> if locally-written transactions must be ignored
*/
public List<Transaction> getTxnsByCommitTimeAscending(
Long fromTimeInclusive,
Long toTimeExclusive,
int count,
List<Long> 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. <tt>null</tt> is allowed.
* @param remoteOnly <tt>true</tt> if locally-written transactions must be ignored
*/
public List<Transaction> getTxnsByCommitTimeDescending(
Long fromTimeInclusive,
Long toTimeExclusive,
int count,
List<Long> excludeTxnIds,
boolean remoteOnly);
/**
* Get a specific list of transactions ordered by commit time.
*
* @param includeTxnIds a list of transaction IDs to search for
* @return Returns the transactions by commit time for the given IDs
*/
public List<Transaction> getTxnsByCommitTimeAscending(List<Long> includeTxnIds);
public int getTxnUpdateCount(Long txnId);
public int getTxnDeleteCount(Long txnId);
public int getTransactionCount();
/**
* @deprecated Replace with query returning a Node object
*/
public List<NodeRef> getTxnChangesForStore(StoreRef storeRef, Long txnId);
/**
* @deprecated Replace with query returning a Node object
*/
public List<NodeRef> getTxnChanges(Long txnId);
public List<Long> getTxnsUnused(Long minTxnId, long maxCommitTime, int count);
public void purgeTxn(Long txnId);
public Long getMinTxnCommitTime();
public Long getMaxTxnCommitTime();
/**
*
* @param parentNodeId
* @param childNodeTypeQNames
* @param value
* @param resultsCallback
*/
public void getChildAssocsByPropertyValue(Long parentNodeId,
QName propertyQName,
Serializable nodeValue,
ChildAssocRefQueryCallback resultsCallback);
}