mirror of
https://github.com/Alfresco/alfresco-community-repo.git
synced 2025-06-30 18:15:39 +00:00
12145: Merged V2.2 to V3.0 (AuthenticationUtil) 12109: AuthenticationUtil and AuthenticationComponent refactor 12152: Removed Lucene usage from lookup of 'sites' root folder 12153: Fix InviteServiceTest by cleaning up leaking authentications 12159: Fix for broken usage pattern of the Threadlocal values in recent AuthenticationUtil refactor. git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@12508 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
584 lines
24 KiB
Java
584 lines
24 KiB
Java
/*
|
|
* Copyright (C) 2005-2007 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.node.db;
|
|
|
|
import java.io.Serializable;
|
|
import java.util.ArrayList;
|
|
import java.util.Date;
|
|
import java.util.List;
|
|
import java.util.Locale;
|
|
import java.util.Map;
|
|
import java.util.Set;
|
|
|
|
import javax.transaction.UserTransaction;
|
|
|
|
import org.alfresco.i18n.I18NUtil;
|
|
import org.alfresco.model.ContentModel;
|
|
import org.alfresco.repo.content.MimetypeMap;
|
|
import org.alfresco.repo.domain.Store;
|
|
import org.alfresco.repo.node.BaseNodeServiceTest;
|
|
import org.alfresco.repo.node.StoreArchiveMap;
|
|
import org.alfresco.repo.node.cleanup.NodeCleanupRegistry;
|
|
import org.alfresco.repo.node.db.NodeDaoService.NodePropertyHandler;
|
|
import org.alfresco.repo.transaction.AlfrescoTransactionSupport;
|
|
import org.alfresco.repo.transaction.RetryingTransactionHelper.RetryingTransactionCallback;
|
|
import org.alfresco.service.cmr.dictionary.DataTypeDefinition;
|
|
import org.alfresco.service.cmr.dictionary.DictionaryService;
|
|
import org.alfresco.service.cmr.repository.ChildAssociationRef;
|
|
import org.alfresco.service.cmr.repository.ContentData;
|
|
import org.alfresco.service.cmr.repository.MLText;
|
|
import org.alfresco.service.cmr.repository.NodeRef;
|
|
import org.alfresco.service.cmr.repository.NodeService;
|
|
import org.alfresco.service.cmr.repository.StoreRef;
|
|
import org.alfresco.service.namespace.QName;
|
|
import org.alfresco.service.namespace.RegexQNamePattern;
|
|
import org.alfresco.service.transaction.TransactionService;
|
|
import org.alfresco.util.Pair;
|
|
import org.apache.commons.lang.mutable.MutableInt;
|
|
|
|
/**
|
|
* @see org.alfresco.repo.node.db.DbNodeServiceImpl
|
|
*
|
|
* @author Derek Hulley
|
|
*/
|
|
@SuppressWarnings("unused")
|
|
public class DbNodeServiceImplTest extends BaseNodeServiceTest
|
|
{
|
|
private TransactionService txnService;
|
|
private NodeDaoService nodeDaoService;
|
|
private DictionaryService dictionaryService;
|
|
|
|
protected NodeService getNodeService()
|
|
{
|
|
// Force cascading
|
|
DbNodeServiceImpl dbNodeServiceImpl = (DbNodeServiceImpl) applicationContext.getBean("dbNodeServiceImpl");
|
|
dbNodeServiceImpl.setCascadeInTransaction(true);
|
|
|
|
return (NodeService) applicationContext.getBean("dbNodeService");
|
|
}
|
|
|
|
@Override
|
|
protected void onSetUpInTransaction() throws Exception
|
|
{
|
|
super.onSetUpInTransaction();
|
|
txnService = (TransactionService) applicationContext.getBean("transactionComponent");
|
|
nodeDaoService = (NodeDaoService) applicationContext.getBean("nodeDaoService");
|
|
dictionaryService = (DictionaryService) applicationContext.getBean("dictionaryService");
|
|
}
|
|
|
|
/**
|
|
* Deletes a child node and then iterates over the children of the parent node,
|
|
* getting the QName. This caused some issues after we did some optimization
|
|
* using lazy loading of the associations.
|
|
*/
|
|
public void testLazyLoadIssue() throws Exception
|
|
{
|
|
Map<QName, ChildAssociationRef> assocRefs = buildNodeGraph();
|
|
// commit results
|
|
setComplete();
|
|
endTransaction();
|
|
|
|
UserTransaction userTransaction = txnService.getUserTransaction();
|
|
|
|
try
|
|
{
|
|
userTransaction.begin();
|
|
|
|
ChildAssociationRef n6pn8Ref = assocRefs.get(QName.createQName(BaseNodeServiceTest.NAMESPACE, "n6_p_n8"));
|
|
NodeRef n6Ref = n6pn8Ref.getParentRef();
|
|
NodeRef n8Ref = n6pn8Ref.getChildRef();
|
|
|
|
// delete n8
|
|
nodeService.deleteNode(n8Ref);
|
|
|
|
// get the parent children
|
|
List<ChildAssociationRef> assocs = nodeService.getChildAssocs(n6Ref);
|
|
for (ChildAssociationRef assoc : assocs)
|
|
{
|
|
// just checking
|
|
}
|
|
|
|
userTransaction.commit();
|
|
}
|
|
catch(Exception e)
|
|
{
|
|
try { userTransaction.rollback(); } catch (IllegalStateException ee) {}
|
|
throw e;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Checks that the node status changes correctly during:
|
|
* <ul>
|
|
* <li>creation</li>
|
|
* <li>property changes</li>
|
|
* <li>aspect changes</li>
|
|
* <li>moving</li>
|
|
* <li>deletion</li>
|
|
* </ul>
|
|
*/
|
|
public void testNodeStatus() throws Throwable
|
|
{
|
|
Map<QName, ChildAssociationRef> assocRefs = buildNodeGraph();
|
|
// get the node to play with
|
|
ChildAssociationRef n6pn8Ref = assocRefs.get(QName.createQName(BaseNodeServiceTest.NAMESPACE, "n6_p_n8"));
|
|
final NodeRef n6Ref = n6pn8Ref.getParentRef();
|
|
final NodeRef n8Ref = n6pn8Ref.getChildRef();
|
|
final Map<QName, Serializable> properties = nodeService.getProperties(n6Ref);
|
|
|
|
// commit results
|
|
setComplete();
|
|
endTransaction();
|
|
|
|
// change property - check status
|
|
RetryingTransactionCallback<Object> changePropertiesWork = new RetryingTransactionCallback<Object>()
|
|
{
|
|
public Object execute()
|
|
{
|
|
nodeService.setProperty(n6Ref, ContentModel.PROP_CREATED, new Date());
|
|
return null;
|
|
}
|
|
};
|
|
executeAndCheck(n6Ref, changePropertiesWork);
|
|
|
|
// add an aspect
|
|
RetryingTransactionCallback<Object> addAspectWork = new RetryingTransactionCallback<Object>()
|
|
{
|
|
public Object execute()
|
|
{
|
|
nodeService.addAspect(n6Ref, ASPECT_QNAME_TEST_MARKER, null);
|
|
return null;
|
|
}
|
|
};
|
|
executeAndCheck(n6Ref, addAspectWork);
|
|
|
|
// remove an aspect
|
|
RetryingTransactionCallback<Object> removeAspectWork = new RetryingTransactionCallback<Object>()
|
|
{
|
|
public Object execute()
|
|
{
|
|
nodeService.removeAspect(n6Ref, ASPECT_QNAME_TEST_MARKER);
|
|
return null;
|
|
}
|
|
};
|
|
executeAndCheck(n6Ref, removeAspectWork);
|
|
|
|
// move the node
|
|
RetryingTransactionCallback<Object> moveNodeWork = new RetryingTransactionCallback<Object>()
|
|
{
|
|
public Object execute()
|
|
{
|
|
nodeService.moveNode(
|
|
n6Ref,
|
|
rootNodeRef,
|
|
ASSOC_TYPE_QNAME_TEST_CHILDREN,
|
|
QName.createQName(NAMESPACE, "moved"));
|
|
return null;
|
|
}
|
|
};
|
|
executeAndCheck(n6Ref, moveNodeWork);
|
|
|
|
// delete the node
|
|
RetryingTransactionCallback<Object> deleteNodeWork = new RetryingTransactionCallback<Object>()
|
|
{
|
|
public Object execute()
|
|
{
|
|
nodeService.deleteNode(n6Ref);
|
|
return null;
|
|
}
|
|
};
|
|
executeAndCheck(n6Ref, deleteNodeWork);
|
|
|
|
// check cascade-deleted nodes
|
|
RetryingTransactionCallback<Object> checkCascadeCallback = new RetryingTransactionCallback<Object>()
|
|
{
|
|
public Object execute()
|
|
{
|
|
// check n6
|
|
NodeRef.Status n6Status = nodeDaoService.getNodeRefStatus(n6Ref);
|
|
if (!n6Status.isDeleted())
|
|
{
|
|
throw new RuntimeException("Deleted node does not have deleted status");
|
|
}
|
|
// n8 is a primary child - it should be deleted too
|
|
NodeRef.Status n8Status = nodeDaoService.getNodeRefStatus(n8Ref);
|
|
if (!n8Status.isDeleted())
|
|
{
|
|
throw new RuntimeException("Cascade-deleted node does not have deleted status");
|
|
}
|
|
return null;
|
|
}
|
|
};
|
|
retryingTransactionHelper.doInTransaction(checkCascadeCallback);
|
|
|
|
// check node recreation
|
|
RetryingTransactionCallback<Object> checkRecreateCallback = new RetryingTransactionCallback<Object>()
|
|
{
|
|
public Object execute()
|
|
{
|
|
properties.put(ContentModel.PROP_STORE_PROTOCOL, n6Ref.getStoreRef().getProtocol());
|
|
properties.put(ContentModel.PROP_STORE_IDENTIFIER, n6Ref.getStoreRef().getIdentifier());
|
|
properties.put(ContentModel.PROP_NODE_UUID, n6Ref.getId());
|
|
|
|
// recreate n6
|
|
nodeService.createNode(
|
|
rootNodeRef,
|
|
ASSOC_TYPE_QNAME_TEST_CHILDREN,
|
|
QName.createQName(NAMESPACE, "recreated-n6"),
|
|
ContentModel.TYPE_CONTAINER,
|
|
properties);
|
|
return null;
|
|
}
|
|
};
|
|
retryingTransactionHelper.doInTransaction(checkRecreateCallback);
|
|
}
|
|
|
|
private void executeAndCheck(NodeRef nodeRef, RetryingTransactionCallback<Object> callback) throws Throwable
|
|
{
|
|
UserTransaction txn = txnService.getUserTransaction();
|
|
txn.begin();
|
|
|
|
NodeRef.Status currentStatus = nodeService.getNodeStatus(nodeRef);
|
|
assertNotNull(currentStatus);
|
|
String currentTxnId = AlfrescoTransactionSupport.getTransactionId();
|
|
assertNotNull(currentTxnId);
|
|
assertNotSame(currentTxnId, currentStatus.getChangeTxnId());
|
|
try
|
|
{
|
|
callback.execute();
|
|
// get the status
|
|
NodeRef.Status newStatus = nodeService.getNodeStatus(nodeRef);
|
|
assertNotNull(newStatus);
|
|
// check
|
|
assertEquals("Change didn't update status", currentTxnId, newStatus.getChangeTxnId());
|
|
txn.commit();
|
|
}
|
|
catch (Throwable e)
|
|
{
|
|
try { txn.rollback(); } catch (Throwable ee) {}
|
|
throw e;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Checks that the string_value retrieval against a property type is working
|
|
*/
|
|
public void testGetContentDataValues() throws Exception
|
|
{
|
|
final DataTypeDefinition contentDataType = dictionaryService.getDataType(DataTypeDefinition.CONTENT);
|
|
|
|
ContentData contentDataSingle = new ContentData("url-single", MimetypeMap.MIMETYPE_TEXT_PLAIN, 0L, null);
|
|
ContentData contentDataMultiple = new ContentData("url-multiple", MimetypeMap.MIMETYPE_TEXT_PLAIN, 0L, null);
|
|
// put this in as a random single property
|
|
nodeService.setProperty(
|
|
rootNodeRef,
|
|
QName.createQName(NAMESPACE, "random-single"),
|
|
contentDataSingle);
|
|
|
|
// create a collection of mixed types
|
|
ArrayList<Serializable> collection = new ArrayList<Serializable>(3);
|
|
collection.add("abc");
|
|
collection.add(new Integer(123));
|
|
collection.add(contentDataMultiple);
|
|
nodeService.setProperty(
|
|
rootNodeRef,
|
|
QName.createQName(NAMESPACE, "random-multiple"),
|
|
collection);
|
|
|
|
// get a list of all content values
|
|
final List<Serializable> allContentDatas = new ArrayList<Serializable>(500);
|
|
NodePropertyHandler handler = new NodePropertyHandler()
|
|
{
|
|
public void handle(NodeRef nodeRef, QName nodeTypeQName, QName propertyQName, Serializable value)
|
|
{
|
|
allContentDatas.add(value);
|
|
}
|
|
};
|
|
nodeDaoService.getPropertyValuesByActualType(contentDataType, handler);
|
|
assertTrue("At least two instances expected", allContentDatas.size() >= 2);
|
|
assertTrue("Single content data not present in results",
|
|
allContentDatas.contains(contentDataSingle));
|
|
assertTrue("Multi-valued buried content data not present in results",
|
|
allContentDatas.contains(contentDataMultiple));
|
|
}
|
|
|
|
public void testMLTextValues() throws Exception
|
|
{
|
|
// Set the server default locale
|
|
Locale.setDefault(Locale.ENGLISH);
|
|
|
|
MLText mlTextProperty = new MLText();
|
|
mlTextProperty.addValue(Locale.ENGLISH, "Very good!");
|
|
mlTextProperty.addValue(Locale.FRENCH, "Très bon!");
|
|
mlTextProperty.addValue(Locale.GERMAN, "Sehr gut!");
|
|
|
|
nodeService.setProperty(
|
|
rootNodeRef,
|
|
BaseNodeServiceTest.PROP_QNAME_ML_TEXT_VALUE,
|
|
mlTextProperty);
|
|
|
|
// Check unfiltered property retrieval
|
|
Serializable textValueDirect = nodeService.getProperty(
|
|
rootNodeRef,
|
|
BaseNodeServiceTest.PROP_QNAME_ML_TEXT_VALUE);
|
|
assertEquals(
|
|
"MLText type not returned direct",
|
|
mlTextProperty,
|
|
textValueDirect);
|
|
|
|
// Check unfiltered mass property retrieval
|
|
Map<QName, Serializable> propertiesDirect = nodeService.getProperties(rootNodeRef);
|
|
assertEquals(
|
|
"MLText type not returned direct in Map",
|
|
mlTextProperty,
|
|
propertiesDirect.get(BaseNodeServiceTest.PROP_QNAME_ML_TEXT_VALUE));
|
|
}
|
|
|
|
/**
|
|
* Ensure that plain strings going into MLText properties is handled
|
|
*/
|
|
@SuppressWarnings("unchecked")
|
|
public void testStringIntoMLTextProperty() throws Exception
|
|
{
|
|
String text = "Hello";
|
|
nodeService.setProperty(rootNodeRef, PROP_QNAME_ML_TEXT_VALUE, text);
|
|
Serializable mlTextCheck = nodeService.getProperty(rootNodeRef, PROP_QNAME_ML_TEXT_VALUE);
|
|
assertTrue("Plain string insertion should be returned as MLText", mlTextCheck instanceof MLText);
|
|
Locale defaultLocale = I18NUtil.getLocale();
|
|
MLText mlTextCheck2 = (MLText) mlTextCheck;
|
|
String mlTextDefaultCheck = mlTextCheck2.getDefaultValue();
|
|
assertEquals("Default MLText value was not set correctly", text, mlTextDefaultCheck);
|
|
|
|
// Reset the property
|
|
nodeService.setProperty(rootNodeRef, PROP_QNAME_ML_TEXT_VALUE, null);
|
|
Serializable nullValueCheck = nodeService.getProperty(rootNodeRef, PROP_QNAME_ML_TEXT_VALUE);
|
|
|
|
// Now, just pass a String in
|
|
nodeService.setProperty(rootNodeRef, PROP_QNAME_ML_TEXT_VALUE, text);
|
|
// Now update the property with some MLText
|
|
MLText mlText = new MLText();
|
|
mlText.addValue(Locale.ENGLISH, "Very good!");
|
|
mlText.addValue(Locale.FRENCH, "Très bon!");
|
|
mlText.addValue(Locale.GERMAN, "Sehr gut!");
|
|
nodeService.setProperty(rootNodeRef, PROP_QNAME_ML_TEXT_VALUE, mlText);
|
|
// Get it back and check
|
|
mlTextCheck = nodeService.getProperty(rootNodeRef, PROP_QNAME_ML_TEXT_VALUE);
|
|
assertEquals("Setting of MLText over String failed.", mlText, mlTextCheck);
|
|
}
|
|
|
|
/**
|
|
* Ensure that plain strings going into MLText properties is handled
|
|
*/
|
|
@SuppressWarnings("unchecked")
|
|
public void testSingleStringMLTextProperty() throws Exception
|
|
{
|
|
// Set the property with single-value MLText
|
|
MLText mlText = new MLText();
|
|
mlText.addValue(Locale.GERMAN, "Sehr gut!");
|
|
nodeService.setProperty(rootNodeRef, PROP_QNAME_ML_TEXT_VALUE, mlText);
|
|
// Get it back and check
|
|
MLText mlTextCheck = (MLText) nodeService.getProperty(rootNodeRef, PROP_QNAME_ML_TEXT_VALUE);
|
|
assertEquals("Setting of MLText over String failed.", mlText, mlTextCheck);
|
|
}
|
|
|
|
public void testDuplicatePrimaryParentHandling() throws Exception
|
|
{
|
|
Map<QName, ChildAssociationRef> assocRefs = buildNodeGraph();
|
|
// get the node to play with
|
|
ChildAssociationRef n1pn3Ref = assocRefs.get(QName.createQName(BaseNodeServiceTest.NAMESPACE, "n1_p_n3"));
|
|
ChildAssociationRef n6pn8Ref = assocRefs.get(QName.createQName(BaseNodeServiceTest.NAMESPACE, "n6_p_n8"));
|
|
final NodeRef n1Ref = n1pn3Ref.getParentRef();
|
|
final NodeRef n8Ref = n6pn8Ref.getChildRef();
|
|
|
|
// Add a make n1 a second primary parent of n8
|
|
Pair<Long, NodeRef> n1Pair = nodeDaoService.getNodePair(n1Ref);
|
|
Pair<Long, NodeRef> n8Pair = nodeDaoService.getNodePair(n8Ref);
|
|
Pair<Long, ChildAssociationRef> assocPair = nodeDaoService.newChildAssoc(
|
|
n1Pair.getFirst(),
|
|
n8Pair.getFirst(),
|
|
true,
|
|
ContentModel.ASSOC_CONTAINS,
|
|
QName.createQName(NAMESPACE, "n1pn8"),
|
|
null);
|
|
|
|
// Now get the node primary parent
|
|
nodeService.getPrimaryParent(n8Ref);
|
|
// Get it again
|
|
nodeService.getPrimaryParent(n8Ref);
|
|
}
|
|
|
|
/**
|
|
* It would appear that an issue has arisen with creating and deleting nodes
|
|
* in the same transaction.
|
|
*/
|
|
public void testInTransactionCreateAndDelete() throws Exception
|
|
{
|
|
// Create a node
|
|
NodeRef nodeRef = nodeService.createNode(
|
|
rootNodeRef,
|
|
ASSOC_TYPE_QNAME_TEST_CHILDREN,
|
|
QName.createQName(NAMESPACE, this.getName()),
|
|
TYPE_QNAME_TEST_CONTENT).getChildRef();
|
|
// Delete the node
|
|
nodeService.deleteNode(nodeRef);
|
|
}
|
|
|
|
public void testCleanup() throws Exception
|
|
{
|
|
@SuppressWarnings("unchecked")
|
|
StoreArchiveMap storeArchiveMap = (StoreArchiveMap) applicationContext.getBean("storeArchiveMap");
|
|
DbNodeServiceImpl ns = (DbNodeServiceImpl) applicationContext.getBean("dbNodeServiceImpl");
|
|
ns.setCascadeInTransaction(false);
|
|
|
|
NodeRef parentNodeRef = nodeService.createNode(
|
|
rootNodeRef,
|
|
ASSOC_TYPE_QNAME_TEST_CHILDREN,
|
|
QName.createQName(NAMESPACE, this.getName()),
|
|
ContentModel.TYPE_FOLDER).getChildRef();
|
|
NodeRef childNodeRef = nodeService.createNode(
|
|
parentNodeRef,
|
|
ContentModel.ASSOC_CONTAINS,
|
|
QName.createQName(NAMESPACE, this.getName()),
|
|
ContentModel.TYPE_FOLDER).getChildRef();
|
|
|
|
// Ensure that the archive feature is enabled
|
|
StoreRef archiveStoreRef = ns.createStore("test", getName() + "-" + System.currentTimeMillis());
|
|
storeArchiveMap.getArchiveMap().put(parentNodeRef.getStoreRef(), archiveStoreRef);
|
|
|
|
// Delete parent. Cascade is OFF, so children should be left in their current store.
|
|
ns.deleteNode(parentNodeRef);
|
|
// Check that the node n1 is in the archive store
|
|
assertFalse("Parent should be deleted", ns.exists(parentNodeRef));
|
|
NodeRef parentArchiveRef = new NodeRef(archiveStoreRef, parentNodeRef.getId());
|
|
assertTrue("Parent should be in the archive store", ns.exists(parentArchiveRef));
|
|
|
|
// Force a commit here
|
|
setComplete();
|
|
endTransaction();
|
|
|
|
NodeCleanupRegistry nodeCleanupRegistry = new NodeCleanupRegistry();
|
|
DbNodeServiceImpl.MoveChildrenToCorrectStore worker = new DbNodeServiceImpl.MoveChildrenToCorrectStore();
|
|
worker.setTransactionService(transactionService);
|
|
worker.setDbNodeService(ns);
|
|
worker.setNodeDaoService(nodeDaoService);
|
|
|
|
// Run cleanup
|
|
worker.doClean();
|
|
}
|
|
|
|
/**
|
|
* Adds a property to a node and checks that it can be found using the low-level DB query
|
|
*/
|
|
public void testGetPropertyValuesByPropertyAndValue() throws Throwable
|
|
{
|
|
String findMeValue = "FIND ME";
|
|
nodeService.setProperty(rootNodeRef, PROP_QNAME_STRING_PROP_SINGLE, findMeValue);
|
|
final MutableInt count = new MutableInt(0);
|
|
// Add a property to the root node and check
|
|
NodePropertyHandler handler = new NodePropertyHandler()
|
|
{
|
|
public void handle(NodeRef nodeRef, QName nodeTypeQName, QName propertyQName, Serializable value)
|
|
{
|
|
if (nodeTypeQName.equals(ContentModel.TYPE_STOREROOT))
|
|
{
|
|
count.setValue(1);
|
|
}
|
|
}
|
|
};
|
|
nodeDaoService.getPropertyValuesByPropertyAndValue(
|
|
rootNodeRef.getStoreRef(),
|
|
PROP_QNAME_STRING_PROP_SINGLE,
|
|
findMeValue,
|
|
handler);
|
|
assertTrue("Set value not found.", count.intValue() == 1);
|
|
}
|
|
|
|
public void testAspectRemovalWithCommit() throws Throwable
|
|
{
|
|
// Create a node to add the aspect to
|
|
NodeRef sourceNodeRef = nodeService.createNode(
|
|
rootNodeRef,
|
|
ASSOC_TYPE_QNAME_TEST_CHILDREN,
|
|
QName.createQName(BaseNodeServiceTest.NAMESPACE, "testAspectRemoval-source"),
|
|
ContentModel.TYPE_CONTAINER).getChildRef();
|
|
|
|
// Create a target for the associations
|
|
NodeRef targetNodeRef = nodeService.createNode(
|
|
rootNodeRef,
|
|
ASSOC_TYPE_QNAME_TEST_CHILDREN,
|
|
QName.createQName(BaseNodeServiceTest.NAMESPACE, "testAspectRemoval-target"),
|
|
ContentModel.TYPE_CONTAINER).getChildRef();
|
|
|
|
// Add the aspect to the source
|
|
nodeService.addAspect(sourceNodeRef, ASPECT_WITH_ASSOCIATIONS, null);
|
|
// Make the associations
|
|
nodeService.addChild(
|
|
sourceNodeRef,
|
|
targetNodeRef,
|
|
ASSOC_ASPECT_CHILD_ASSOC,
|
|
QName.createQName(NAMESPACE, "aspect-child"));
|
|
nodeService.createAssociation(sourceNodeRef, targetNodeRef, ASSOC_ASPECT_NORMAL_ASSOC);
|
|
|
|
// Check that the correct associations are present
|
|
assertEquals("Expected exactly one child",
|
|
1, nodeService.getChildAssocs(sourceNodeRef).size());
|
|
assertEquals("Expected exactly one target",
|
|
1, nodeService.getTargetAssocs(sourceNodeRef, RegexQNamePattern.MATCH_ALL).size());
|
|
|
|
// Force a commit here
|
|
setComplete();
|
|
endTransaction();
|
|
|
|
// start another transaction to remove the aspect
|
|
UserTransaction txn = txnService.getUserTransaction();
|
|
txn.begin();
|
|
|
|
try
|
|
{
|
|
Set<QName> aspects = nodeService.getAspects(sourceNodeRef);
|
|
int noAspectsBefore = aspects.size();
|
|
|
|
// Now remove the aspect
|
|
nodeService.removeAspect(sourceNodeRef, ASPECT_WITH_ASSOCIATIONS);
|
|
|
|
// Check that the associations were removed
|
|
assertEquals("Expected exactly zero child",
|
|
0, nodeService.getChildAssocs(sourceNodeRef).size());
|
|
assertEquals("Expected exactly zero target",
|
|
0, nodeService.getTargetAssocs(sourceNodeRef, RegexQNamePattern.MATCH_ALL).size());
|
|
aspects = nodeService.getAspects(sourceNodeRef);
|
|
assertEquals("Expected exactly one less aspect",
|
|
noAspectsBefore-1, aspects.size());
|
|
|
|
txn.commit();
|
|
}
|
|
catch (Throwable e)
|
|
{
|
|
try { txn.rollback(); } catch (Throwable ee) {}
|
|
throw e;
|
|
}
|
|
}
|
|
}
|