/* * 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 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 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: * */ public void testNodeStatus() throws Throwable { Map 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 properties = nodeService.getProperties(n6Ref); // commit results setComplete(); endTransaction(); // change property - check status RetryingTransactionCallback changePropertiesWork = new RetryingTransactionCallback() { public Object execute() { nodeService.setProperty(n6Ref, ContentModel.PROP_CREATED, new Date()); return null; } }; executeAndCheck(n6Ref, changePropertiesWork); // add an aspect RetryingTransactionCallback addAspectWork = new RetryingTransactionCallback() { public Object execute() { nodeService.addAspect(n6Ref, ASPECT_QNAME_TEST_MARKER, null); return null; } }; executeAndCheck(n6Ref, addAspectWork); // remove an aspect RetryingTransactionCallback removeAspectWork = new RetryingTransactionCallback() { public Object execute() { nodeService.removeAspect(n6Ref, ASPECT_QNAME_TEST_MARKER); return null; } }; executeAndCheck(n6Ref, removeAspectWork); // move the node RetryingTransactionCallback moveNodeWork = new RetryingTransactionCallback() { 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 deleteNodeWork = new RetryingTransactionCallback() { public Object execute() { nodeService.deleteNode(n6Ref); return null; } }; executeAndCheck(n6Ref, deleteNodeWork); // check cascade-deleted nodes RetryingTransactionCallback checkCascadeCallback = new RetryingTransactionCallback() { 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 checkRecreateCallback = new RetryingTransactionCallback() { 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 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 collection = new ArrayList(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 allContentDatas = new ArrayList(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 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 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 n1Pair = nodeDaoService.getNodePair(n1Ref); Pair n8Pair = nodeDaoService.getNodePair(n8Ref); Pair 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 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; } } }