mirror of
				https://github.com/Alfresco/alfresco-community-repo.git
				synced 2025-10-29 15:21:53 +00:00 
			
		
		
		
	- incl. taking Hibernate libs from 3.1 and adding missing file from earlier merge
   13321: Fix ETHREEOH-1407: System error occur during "Undo Selected" action if no items are selected
   13322: ETHREEOH-1206: Throwing Alfresco Exception on OnUpdateProperties behaviour resets Encoding field to Big (first entry)
   13326: (RECORD ONLY) Removed 'dev' from version label.
   13330: Fix ETHREEOH-1408: Incorrect button name at "Manage Task: Submitted" page
   13337: Fix for ETHREEOH-1409 and further fix for ETHREEOH-1408
   13338: Removed svn:mergeinfo
   13346: Make startup bat script check JAVA can be found.
   13351: ETHREEOH-1386 validate ASR and FSR hostname.
   13359: ETHREEOH-1435: Share doesn't extract document metadata correctly
   13360: Fix ETHREEOH-821: SDK dependencies
   13369: Fixed distribute-sdk target for when it's run locally
   13382: ETHREEOH-1437: Container creation induces an unexpected permission allocation in Share
   13391: Shutdown backstop continues if logging throws errors.
   13394: Fix ETHREEOH-1457 - MT coci issue with bootstrap (eg. data dictionary) content
   13400: Activate JAWS-223: Adobe LC Hibernate Dialect Loading
   13401: Support for JAWS-215, mysql and oracle
   13413: Fix ETHREEOH-1458 - MT delete->archive
   ___________________________________________________________________
   Modified: svn:mergeinfo
      Merged /alfresco/BRANCHES/V3.1:r
                           13321-13322,13326-13327,13330,13337-13339,13341-13347,13351,13354-13355,13358-13363,
                           13365,13367,13369,13382,13385-13392,13394,13400-13403,13405-13406,13413
git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@13617 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
		
	
		
			
				
	
	
		
			583 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
			
		
		
	
	
			583 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
| /*
 | |
|  * Copyright (C) 2005-2009 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.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.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;
 | |
|        }
 | |
|     }
 | |
| }
 |