mirror of
				https://github.com/Alfresco/alfresco-community-repo.git
				synced 2025-10-15 15:02:20 +00:00 
			
		
		
		
	107541: Merged 5.0.N (5.0.3) to HEAD-BUG-FIX (5.1/Cloud) (PARTIAL MERGE)
      107413: Merged DEV to 5.0.N (5.0.3)
         106858 : MNT-13545: JavaDoc : Inconsistencies between the Java doc and the actual code
            - Cleaning of Javadoc,
   107565: MNT-13545 Fix compilation after merge of Javadoc
git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@107633 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
		
	
		
			
				
	
	
		
			1457 lines
		
	
	
		
			62 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
			
		
		
	
	
			1457 lines
		
	
	
		
			62 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
| /*
 | |
|  * Copyright (C) 2005-2013 Alfresco Software Limited.
 | |
|  *
 | |
|  * This file is part of Alfresco
 | |
|  *
 | |
|  * Alfresco is free software: you can redistribute it and/or modify
 | |
|  * it under the terms of the GNU Lesser General Public License as published by
 | |
|  * the Free Software Foundation, either version 3 of the License, or
 | |
|  * (at your option) any later version.
 | |
|  *
 | |
|  * Alfresco 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 Lesser General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU Lesser General Public License
 | |
|  * along with Alfresco. If not, see <http://www.gnu.org/licenses/>.
 | |
|  */
 | |
| package org.alfresco.repo.coci;
 | |
| 
 | |
| import java.io.Serializable;
 | |
| import java.util.Collections;
 | |
| import java.util.Date;
 | |
| import java.util.HashMap;
 | |
| import java.util.List;
 | |
| import java.util.Locale;
 | |
| import java.util.Map;
 | |
| 
 | |
| import net.sf.acegisecurity.AccessDeniedException;
 | |
| 
 | |
| import org.alfresco.model.ContentModel;
 | |
| import org.alfresco.repo.content.MimetypeMap;
 | |
| import org.alfresco.repo.node.integrity.IntegrityException;
 | |
| import org.alfresco.repo.security.authentication.AuthenticationComponent;
 | |
| import org.alfresco.repo.security.authentication.AuthenticationUtil;
 | |
| import org.alfresco.repo.security.authentication.AuthenticationUtil.RunAsWork;
 | |
| import org.alfresco.repo.transaction.RetryingTransactionHelper.RetryingTransactionCallback;
 | |
| import org.alfresco.repo.version.VersionModel;
 | |
| import org.alfresco.service.ServiceRegistry;
 | |
| import org.alfresco.service.cmr.coci.CheckOutCheckInService;
 | |
| import org.alfresco.service.cmr.lock.LockService;
 | |
| import org.alfresco.service.cmr.lock.LockType;
 | |
| import org.alfresco.service.cmr.lock.NodeLockedException;
 | |
| import org.alfresco.service.cmr.model.FileFolderService;
 | |
| import org.alfresco.service.cmr.repository.AssociationRef;
 | |
| import org.alfresco.service.cmr.repository.ChildAssociationRef;
 | |
| import org.alfresco.service.cmr.repository.ContentData;
 | |
| import org.alfresco.service.cmr.repository.ContentReader;
 | |
| import org.alfresco.service.cmr.repository.ContentService;
 | |
| import org.alfresco.service.cmr.repository.ContentWriter;
 | |
| import org.alfresco.service.cmr.repository.CopyService;
 | |
| 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.cmr.security.MutableAuthenticationService;
 | |
| import org.alfresco.service.cmr.security.PermissionService;
 | |
| import org.alfresco.service.cmr.security.PersonService;
 | |
| import org.alfresco.service.cmr.version.Version;
 | |
| import org.alfresco.service.cmr.version.VersionService;
 | |
| import org.alfresco.service.cmr.version.VersionType;
 | |
| import org.alfresco.service.namespace.NamespaceService;
 | |
| import org.alfresco.service.namespace.QName;
 | |
| import org.alfresco.service.transaction.TransactionService;
 | |
| import org.alfresco.test_category.BaseSpringTestsCategory;
 | |
| import org.alfresco.test_category.OwnJVMTestsCategory;
 | |
| import org.alfresco.util.BaseSpringTest;
 | |
| import org.alfresco.util.GUID;
 | |
| import org.alfresco.util.PropertyMap;
 | |
| import org.alfresco.util.TestWithUserUtils;
 | |
| import org.junit.experimental.categories.Category;
 | |
| import org.springframework.extensions.surf.util.I18NUtil;
 | |
| 
 | |
| /**
 | |
|  * Version operations service implementation unit tests
 | |
|  * 
 | |
|  * @author Roy Wetherall
 | |
|  */
 | |
| @Category(BaseSpringTestsCategory.class)
 | |
| public class CheckOutCheckInServiceImplTest extends BaseSpringTest 
 | |
| {
 | |
|     /**
 | |
|      * Services used by the tests
 | |
|      */
 | |
|     private NodeService nodeService;
 | |
|     private CheckOutCheckInService cociService;
 | |
|     private ContentService contentService;
 | |
|     private VersionService versionService;
 | |
|     private MutableAuthenticationService authenticationService;
 | |
|     private LockService lockService;
 | |
|     private TransactionService transactionService;
 | |
|     private PermissionService permissionService;
 | |
|     private CopyService copyService;
 | |
|     private PersonService personService;
 | |
|     private FileFolderService fileFolderService;
 | |
| 
 | |
|     /**
 | |
|      * Data used by the tests
 | |
|      */
 | |
|     private StoreRef storeRef;
 | |
|     private NodeRef rootNodeRef;    
 | |
|     private NodeRef nodeRef;
 | |
|     private String userNodeRef;
 | |
|     private NodeRef folderNodeRef;
 | |
|     private NodeRef fileNodeRef;
 | |
|     
 | |
|     /**
 | |
|      * Types and properties used by the tests
 | |
|      */
 | |
|     private static final String TEST_VALUE_NAME = "myDocument.doc";
 | |
|     private static final String TEST_VALUE_2 = "testValue2";
 | |
|     private static final String TEST_VALUE_3 = "testValue3";
 | |
|     private static final QName PROP_NAME_QNAME = QName.createQName(NamespaceService.CONTENT_MODEL_1_0_URI, "name");
 | |
|     private static final QName PROP2_QNAME = ContentModel.PROP_DESCRIPTION;
 | |
|     private static final String CONTENT_1 = "This is some content";
 | |
|     private static final String CONTENT_2 = "This is the cotent modified.";
 | |
|     
 | |
|     /**
 | |
|      * User details 
 | |
|      */
 | |
|     //private static final String USER_NAME = "cociTest" + GUID.generate();
 | |
|     private String userName;
 | |
|     private static final String PWD = "password";
 | |
|     
 | |
|     /**
 | |
|      * On setup in transaction implementation
 | |
|      */
 | |
|     @Override
 | |
|     protected void onSetUpInTransaction() 
 | |
|         throws Exception 
 | |
|     {
 | |
|         // Set the services
 | |
|         this.cociService = (CheckOutCheckInService)this.applicationContext.getBean("checkOutCheckInService");
 | |
|         this.contentService = (ContentService)this.applicationContext.getBean("contentService");
 | |
|         this.versionService = (VersionService)this.applicationContext.getBean("versionService");
 | |
|         this.authenticationService = (MutableAuthenticationService)this.applicationContext.getBean("authenticationService");
 | |
|         this.lockService = (LockService)this.applicationContext.getBean("lockService");
 | |
|         this.transactionService = (TransactionService)this.applicationContext.getBean("transactionComponent");
 | |
|         this.permissionService = (PermissionService)this.applicationContext.getBean("permissionService");
 | |
|         this.copyService = (CopyService)this.applicationContext.getBean("copyService");
 | |
|         this.personService = (PersonService) this.applicationContext.getBean("PersonService");
 | |
|         ServiceRegistry serviceRegistry = (ServiceRegistry) this.applicationContext.getBean("ServiceRegistry");
 | |
|         this.fileFolderService = serviceRegistry.getFileFolderService();
 | |
|         this.nodeService = serviceRegistry.getNodeService();
 | |
|         
 | |
|         // Authenticate as system to create initial test data set
 | |
|         AuthenticationComponent authenticationComponent = (AuthenticationComponent)this.applicationContext.getBean("authenticationComponent");
 | |
|         authenticationComponent.setSystemUserAsCurrentUser();
 | |
|     
 | |
|         // Create the store and get the root node reference
 | |
|         this.storeRef = nodeService.createStore(StoreRef.PROTOCOL_WORKSPACE, "Test_" + System.currentTimeMillis());
 | |
|         this.rootNodeRef = nodeService.getRootNode(storeRef);
 | |
|         
 | |
|         // Create the node used for tests
 | |
|         ChildAssociationRef childAssocRef = nodeService.createNode(
 | |
|                 rootNodeRef,
 | |
|                 ContentModel.ASSOC_CHILDREN,
 | |
|                 QName.createQName("test"),
 | |
|                 ContentModel.TYPE_CONTENT);
 | |
|         this.nodeRef = childAssocRef.getChildRef();
 | |
|         nodeService.addAspect(this.nodeRef, ContentModel.ASPECT_TITLED, null);
 | |
|         nodeService.setProperty(this.nodeRef, ContentModel.PROP_NAME, TEST_VALUE_NAME);
 | |
|         nodeService.setProperty(this.nodeRef, PROP2_QNAME, TEST_VALUE_2);
 | |
|         
 | |
|         // Add the initial content to the node
 | |
|         ContentWriter contentWriter = this.contentService.getWriter(this.nodeRef, ContentModel.PROP_CONTENT, true);
 | |
|         contentWriter.setMimetype("text/plain");
 | |
|         contentWriter.setEncoding("UTF-8");
 | |
|         contentWriter.putContent(CONTENT_1);
 | |
|         
 | |
|         // Add the lock and version aspects to the created node
 | |
|         nodeService.addAspect(this.nodeRef, ContentModel.ASPECT_VERSIONABLE, null);
 | |
|         nodeService.addAspect(this.nodeRef, ContentModel.ASPECT_LOCKABLE, null);        
 | |
|         
 | |
|         // Create and authenticate the user
 | |
|         this.userName = "cociTest" + GUID.generate();
 | |
|         TestWithUserUtils.createUser(this.userName, PWD, this.rootNodeRef, this.nodeService, this.authenticationService);
 | |
|         TestWithUserUtils.authenticateUser(this.userName, PWD, this.rootNodeRef, this.authenticationService);
 | |
|         this.userNodeRef = TestWithUserUtils.getCurrentUser(this.authenticationService);
 | |
|         
 | |
|         permissionService.setPermission(this.rootNodeRef, this.userName, PermissionService.ALL_PERMISSIONS, true);
 | |
|         permissionService.setPermission(this.nodeRef, this.userName, PermissionService.ALL_PERMISSIONS, true);
 | |
| 
 | |
|         folderNodeRef = nodeService.createNode(
 | |
|                 rootNodeRef,
 | |
|                 ContentModel.ASSOC_CHILDREN,
 | |
|                 QName.createQName("test"),
 | |
|                 ContentModel.TYPE_FOLDER,
 | |
|                 Collections.<QName, Serializable>singletonMap(ContentModel.PROP_NAME, "folder")).getChildRef();
 | |
|         fileNodeRef = nodeService.createNode(
 | |
|                 folderNodeRef,
 | |
|                 ContentModel.ASSOC_CONTAINS,
 | |
|                 QName.createQName("test"),
 | |
|                 ContentModel.TYPE_CONTENT,
 | |
|                 Collections.<QName, Serializable>singletonMap(ContentModel.PROP_NAME, "file")).getChildRef();
 | |
|         contentWriter = this.contentService.getWriter(fileNodeRef, ContentModel.PROP_CONTENT, true);
 | |
|         contentWriter.setMimetype("text/plain");
 | |
|         contentWriter.setEncoding("UTF-8");
 | |
|         contentWriter.putContent(CONTENT_1);
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Helper method that creates a bag of properties for the test type
 | |
|      * 
 | |
|      * @return  bag of properties
 | |
|      */
 | |
|     private Map<QName, Serializable> createTypePropertyBag()
 | |
|     {
 | |
|         Map<QName, Serializable> result = new HashMap<QName, Serializable>();
 | |
|         result.put(PROP_NAME_QNAME, TEST_VALUE_NAME);
 | |
|         return result;
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Test checkout 
 | |
|      */
 | |
|     public void testCheckOut()
 | |
|     {
 | |
|         checkout();
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * 
 | |
|      * @return NodeRef
 | |
|      */
 | |
|     private NodeRef checkout()
 | |
|     {
 | |
|         // Check out the node
 | |
|         NodeRef workingCopy = cociService.checkout(
 | |
|                 this.nodeRef, 
 | |
|                 this.rootNodeRef, 
 | |
|                 ContentModel.ASSOC_CHILDREN, 
 | |
|                 QName.createQName("workingCopy"));
 | |
|         assertNotNull(workingCopy);
 | |
|         
 | |
|         //System.out.println(NodeStoreInspector.dumpNodeStore(this.nodeService, this.storeRef));
 | |
|         
 | |
|         // Ensure that the working copy and copy aspect has been applied
 | |
|         assertTrue(nodeService.hasAspect(workingCopy, ContentModel.ASPECT_WORKING_COPY));    
 | |
|         assertTrue(nodeService.hasAspect(workingCopy, ContentModel.ASPECT_COPIEDFROM));
 | |
|         
 | |
|         // Check that the working copy owner has been set correctly
 | |
|         assertEquals(this.userNodeRef, nodeService.getProperty(workingCopy, ContentModel.PROP_WORKING_COPY_OWNER));
 | |
| 
 | |
|         
 | |
|         // Check that the working copy name has been set correctly
 | |
|         String name = (String)this.nodeService.getProperty(this.nodeRef, PROP_NAME_QNAME);
 | |
|         String expectedWorkingCopyLabel = I18NUtil.getMessage("coci_service.working_copy_label");
 | |
|         String expectedWorkingCopyName = CheckOutCheckInServiceImpl.createWorkingCopyName(name, expectedWorkingCopyLabel);
 | |
|         String workingCopyName = (String)this.nodeService.getProperty(workingCopy, PROP_NAME_QNAME);
 | |
|         assertEquals(expectedWorkingCopyName, workingCopyName);
 | |
|         // Check a record has been kept of the working copy label used to create the working copy name
 | |
|         assertEquals(
 | |
|                     "No record of working copy label kept",
 | |
|                     expectedWorkingCopyLabel,
 | |
|                     nodeService.getProperty(workingCopy, ContentModel.PROP_WORKING_COPY_LABEL));
 | |
|         
 | |
|         // Ensure that the content has been copied correctly
 | |
|         ContentReader contentReader = this.contentService.getReader(this.nodeRef, ContentModel.PROP_CONTENT);
 | |
|         assertNotNull(contentReader);
 | |
|         ContentReader contentReader2 = this.contentService.getReader(workingCopy, ContentModel.PROP_CONTENT);
 | |
|         assertNotNull(contentReader2);
 | |
|         assertEquals(
 | |
|                 "The content string of the working copy should match the original immediatly after checkout.", 
 | |
|                 contentReader.getContentString(), 
 | |
|                 contentReader2.getContentString());
 | |
|         
 | |
|         return workingCopy;
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Test checkIn
 | |
|      */
 | |
|     public void testCheckIn()
 | |
|     {
 | |
|         NodeRef workingCopy = checkout();
 | |
|         
 | |
|         // Test standard check-in
 | |
|         Map<String, Serializable> versionProperties = new HashMap<String, Serializable>();
 | |
|         versionProperties.put(Version.PROP_DESCRIPTION, "This is a test version");        
 | |
|         cociService.checkin(workingCopy, versionProperties);    
 | |
|         
 | |
|         // Test check-in with content
 | |
|         NodeRef workingCopy3 = checkout();
 | |
|         
 | |
|         nodeService.setProperty(workingCopy3, PROP_NAME_QNAME, TEST_VALUE_2);
 | |
|         nodeService.setProperty(workingCopy3, PROP2_QNAME, TEST_VALUE_3);
 | |
|         ContentWriter tempWriter = this.contentService.getWriter(workingCopy3, ContentModel.PROP_CONTENT, false);
 | |
|         assertNotNull(tempWriter);
 | |
|         tempWriter.putContent(CONTENT_2);
 | |
|         String contentUrl = tempWriter.getContentUrl();
 | |
|         Map<String, Serializable> versionProperties3 = new HashMap<String, Serializable>();
 | |
|         versionProperties3.put(Version.PROP_DESCRIPTION, "description");
 | |
|         versionProperties3.put(VersionModel.PROP_VERSION_TYPE, VersionType.MAJOR);
 | |
|         NodeRef origNodeRef = cociService.checkin(workingCopy3, versionProperties3, contentUrl, true);
 | |
|         assertNotNull(origNodeRef);
 | |
|         
 | |
|         // Check the checked in content
 | |
|         ContentReader contentReader = this.contentService.getReader(origNodeRef, ContentModel.PROP_CONTENT);
 | |
|         assertNotNull(contentReader);
 | |
|         assertEquals(CONTENT_2, contentReader.getContentString());
 | |
|         
 | |
|         // Check that the version history is correct
 | |
|         Version version = this.versionService.getCurrentVersion(origNodeRef);
 | |
|         assertNotNull(version);
 | |
|         assertEquals("description", version.getDescription());
 | |
|         assertEquals(VersionType.MAJOR, version.getVersionType());
 | |
|         NodeRef versionNodeRef = version.getFrozenStateNodeRef();
 | |
|         assertNotNull(versionNodeRef);
 | |
|         
 | |
|         // Check the verioned content
 | |
|         ContentReader versionContentReader = this.contentService.getReader(versionNodeRef, ContentModel.PROP_CONTENT);
 | |
|         assertNotNull(versionContentReader);    
 | |
|         assertEquals(CONTENT_2, versionContentReader.getContentString());
 | |
|         
 | |
|         // Check that the name is not updated during the check-in
 | |
|         assertEquals(TEST_VALUE_2, nodeService.getProperty(versionNodeRef, PROP_NAME_QNAME));
 | |
|         assertEquals(TEST_VALUE_2, nodeService.getProperty(origNodeRef, PROP_NAME_QNAME));
 | |
|         
 | |
|         // Check that the other properties are updated during the check-in
 | |
|         assertEquals(TEST_VALUE_3, nodeService.getProperty(versionNodeRef, PROP2_QNAME));
 | |
|         assertEquals(TEST_VALUE_3, nodeService.getProperty(origNodeRef, PROP2_QNAME));
 | |
|         
 | |
|         // Cancel the check out after is has been left checked out
 | |
|         cociService.cancelCheckout(workingCopy3);
 | |
|         
 | |
|         // Test keep checked out flag
 | |
|         NodeRef workingCopy2 = checkout();        
 | |
|         Map<String, Serializable> versionProperties2 = new HashMap<String, Serializable>();
 | |
|         versionProperties2.put(Version.PROP_DESCRIPTION, "Another version test");        
 | |
|         this.cociService.checkin(workingCopy2, versionProperties2, null, true);
 | |
|         this.cociService.checkin(workingCopy2, new HashMap<String, Serializable>(), null, true);    
 | |
|     }
 | |
| 
 | |
|     public void testCheckInVersionedNode_MNT_8789()
 | |
|     {
 | |
|         String versionDescription = "This is a test version";
 | |
| 
 | |
|         // Create a node as the "A" user
 | |
|         NodeRef nodeA = AuthenticationUtil.runAs(new AuthenticationUtil.RunAsWork<NodeRef>()
 | |
|         {
 | |
|             @Override
 | |
|             public NodeRef doWork() throws Exception
 | |
|             {
 | |
|                 return transactionService.getRetryingTransactionHelper().doInTransaction(new RetryingTransactionCallback<NodeRef>()
 | |
|                 {
 | |
|                     public NodeRef execute() throws Exception
 | |
|                     {
 | |
|                         AuthenticationUtil.setFullyAuthenticatedUser(userName);
 | |
|                         NodeRef a = nodeService.createNode(
 | |
|                                 rootNodeRef,
 | |
|                                 ContentModel.ASSOC_CONTAINS,
 | |
|                                 QName.createQName("{test}NodeForA"),
 | |
|                                 ContentModel.TYPE_CONTENT
 | |
|                         ).getChildRef();
 | |
|                         nodeService.addAspect(a, ContentModel.ASPECT_AUDITABLE, null);
 | |
|                         nodeService.addAspect(a, ContentModel.ASPECT_VERSIONABLE, null);
 | |
|                         return a;
 | |
|                     }
 | |
|                 }
 | |
|                 );
 | |
|             }
 | |
|         }, this.userName);
 | |
| 
 | |
|         // Check that it's owned and modified by test user
 | |
|         assertEquals(this.userName, nodeService.getProperty(nodeA, ContentModel.PROP_CREATOR));
 | |
|         assertEquals(this.userName, nodeService.getProperty(nodeA, ContentModel.PROP_MODIFIER));
 | |
|         assertEquals(true, nodeService.hasAspect(nodeA, ContentModel.ASPECT_VERSIONABLE));
 | |
| 
 | |
|         // Checkout and check in by admin
 | |
|         AuthenticationUtil.setFullyAuthenticatedUser(AuthenticationUtil.getAdminUserName());
 | |
|         NodeRef workingCopy = cociService.checkout(nodeA);
 | |
|         Map<String, Serializable> versionProperties = new HashMap<String, Serializable>();
 | |
|         versionProperties.put(Version.PROP_DESCRIPTION, versionDescription);
 | |
|         cociService.checkin(workingCopy, versionProperties);
 | |
| 
 | |
|         // Ensure it's still owned by test user
 | |
|         assertEquals(this.userName, nodeService.getProperty(nodeA, ContentModel.PROP_CREATOR));
 | |
|         // Modified by admin, but as nothing changed, test user will be put into version
 | |
|         assertEquals(this.userName, nodeService.getProperty(nodeA, ContentModel.PROP_MODIFIER));
 | |
|         assertEquals(true, nodeService.hasAspect(nodeA, ContentModel.ASPECT_VERSIONABLE));
 | |
|         // Save the modified date
 | |
|         Serializable modifiedDate = nodeService.getProperty(nodeA, ContentModel.PROP_MODIFIED);
 | |
| 
 | |
|         // Now check the version
 | |
|         Version version = this.versionService.getCurrentVersion(nodeA);
 | |
|         assertNotNull(version);
 | |
|         assertEquals(versionDescription, version.getDescription());
 | |
|         // Admin checked in the node, but as the working copy was not modified, the modifier should not change
 | |
|         assertEquals(this.userName, version.getFrozenModifier());
 | |
|         // The date should NOT have changed, as nothing was changed in the working copy
 | |
|         assertEquals(true, version.getFrozenModifiedDate().equals(modifiedDate));
 | |
|         NodeRef versionNodeRef = version.getFrozenStateNodeRef();
 | |
|         assertNotNull(versionNodeRef);
 | |
| 
 | |
|         nodeService.deleteNode(nodeA);
 | |
|         AuthenticationUtil.setFullyAuthenticatedUser(this.userName);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * <a href="https://issues.alfresco.com/jira/browse/MNT-9202">MNT-9202</a>
 | |
|      */
 | |
|     public void testDeleteSourceOfLockedCopy()
 | |
|     {
 | |
|         // Create a FolderA
 | |
|         final NodeRef folderA = transactionService.getRetryingTransactionHelper().doInTransaction(new RetryingTransactionCallback<NodeRef>()
 | |
|         {
 | |
|             public NodeRef execute() throws Exception
 | |
|             {
 | |
|                 NodeRef a = nodeService.createNode(
 | |
|                         rootNodeRef,
 | |
|                         ContentModel.ASSOC_CONTAINS,
 | |
|                         QName.createQName("{test}FolderA"),
 | |
|                         ContentModel.TYPE_FOLDER
 | |
|                 ).getChildRef();
 | |
|                 return a;
 | |
|             }
 | |
|         });
 | |
|         // Create a FolderB
 | |
|         NodeRef folderB = transactionService.getRetryingTransactionHelper().doInTransaction(new RetryingTransactionCallback<NodeRef>()
 | |
|         {
 | |
|             public NodeRef execute() throws Exception
 | |
|             {
 | |
|                 NodeRef b = nodeService.createNode(
 | |
|                         rootNodeRef,
 | |
|                         ContentModel.ASSOC_CONTAINS,
 | |
|                         QName.createQName("{test}FolderB"),
 | |
|                         ContentModel.TYPE_FOLDER
 | |
|                 ).getChildRef();
 | |
|                 return b;
 | |
|             }
 | |
|         });
 | |
|         // Create content in FolderA
 | |
|         NodeRef file = transactionService.getRetryingTransactionHelper().doInTransaction(new RetryingTransactionCallback<NodeRef>()
 | |
|         {
 | |
|             public NodeRef execute() throws Exception
 | |
|             {
 | |
|                 NodeRef file = nodeService.createNode(
 | |
|                         folderA,
 | |
|                         ContentModel.ASSOC_CONTAINS,
 | |
|                         QName.createQName("{test}file"),
 | |
|                         ContentModel.TYPE_CONTENT
 | |
|                 ).getChildRef();
 | |
|                 return file;
 | |
|             }
 | |
|         });
 | |
|         // Copy the file to FolderB
 | |
|         NodeRef copy = this.copyService.copy(
 | |
|                 file,
 | |
|                 folderB,
 | |
|                 ContentModel.ASSOC_CHILDREN,
 | |
|                 QName.createQName("{test}copy"),
 | |
|                 true);
 | |
| 
 | |
|         // Check out the copy
 | |
|         NodeRef workingCopy = this.cociService.checkout(copy);
 | |
| 
 | |
|         assertNotNull(workingCopy);
 | |
|         Date wcModBefore = (Date) nodeService.getProperty(workingCopy, ContentModel.PROP_MODIFIED);
 | |
| 
 | |
|         // Allow a second to pass so that we will detect any change to the cm:modified time
 | |
|         synchronized(this)
 | |
|         {
 | |
|             try { this.wait(1000L); } catch (InterruptedException e) {}
 | |
|         }
 | |
|         
 | |
|         // Try to delete the original file
 | |
|         this.nodeService.deleteNode(file);
 | |
|         // That worked.  Check the date.
 | |
|         Date wcModAfter = (Date) nodeService.getProperty(workingCopy, ContentModel.PROP_MODIFIED);
 | |
|         assertEquals("cm:modified should not change on the copied node when deleting the original", wcModBefore, wcModAfter);
 | |
|     }
 | |
| 
 | |
|     public void testCheckOutCheckInWithDifferentLocales()
 | |
|     {
 | |
|         // Check-out nodeRef using the locale fr_FR
 | |
|         Locale.setDefault(Locale.FRANCE);
 | |
|         NodeRef workingCopy = this.cociService.checkout(
 | |
|                 this.nodeRef, 
 | |
|                 this.rootNodeRef, 
 | |
|                 ContentModel.ASSOC_CHILDREN, 
 | |
|                 QName.createQName("workingCopy"));
 | |
|         assertNotNull(workingCopy);
 | |
|         
 | |
|         // Check that the working copy name has been set correctly
 | |
|         String workingCopyName = (String) nodeService.getProperty(workingCopy, PROP_NAME_QNAME);
 | |
|         assertEquals("Working copy name not correct", "myDocument (Copie de Travail).doc", workingCopyName);
 | |
|         
 | |
|         // Check-in using the locale en_GB
 | |
|         Locale.setDefault(Locale.UK);
 | |
|         Map<String, Serializable> versionProperties = new HashMap<String, Serializable>();
 | |
|         versionProperties.put(Version.PROP_DESCRIPTION, "This is a test version");      
 | |
|         cociService.checkin(workingCopy, versionProperties);
 | |
|         
 | |
|         String name = (String) nodeService.getProperty(nodeRef, PROP_NAME_QNAME);
 | |
|         assertEquals("Working copy label was not removed.", "myDocument.doc", name);
 | |
|     }
 | |
|     
 | |
|     public void testCheckOutCheckInWithAlteredWorkingCopyName()
 | |
|     {
 | |
|         // Check-out nodeRef using the locale fr_FR
 | |
|         Locale.setDefault(Locale.FRANCE);
 | |
|         NodeRef workingCopy = this.cociService.checkout(
 | |
|                 this.nodeRef, 
 | |
|                 this.rootNodeRef, 
 | |
|                 ContentModel.ASSOC_CHILDREN, 
 | |
|                 QName.createQName("workingCopy"));
 | |
|         assertNotNull(workingCopy);
 | |
|         
 | |
|         // Check that the working copy name has been set correctly
 | |
|         String workingCopyName = (String) nodeService.getProperty(workingCopy, PROP_NAME_QNAME);
 | |
|         assertEquals("Working copy name not correct", "myDocument (Copie de Travail).doc", workingCopyName);
 | |
|         
 | |
|         // Alter the working copy name
 | |
|         nodeService.setProperty(workingCopy, PROP_NAME_QNAME, "newName (Copie de Travail).doc");
 | |
|         
 | |
|         // Check-in using the locale en_GB
 | |
|         Locale.setDefault(Locale.UK);
 | |
|         Map<String, Serializable> versionProperties = new HashMap<String, Serializable>();
 | |
|         versionProperties.put(Version.PROP_DESCRIPTION, "This is a test version");      
 | |
|         cociService.checkin(workingCopy, versionProperties);
 | |
|         
 | |
|         String name = (String) nodeService.getProperty(nodeRef, PROP_NAME_QNAME);
 | |
|         assertEquals("File not renamed correctly.", "newName.doc", name);
 | |
|     }
 | |
|     
 | |
|     public void testCheckInWithNameChange()
 | |
|     {
 | |
|         // Check out the file
 | |
|         NodeRef fileWorkingCopyNodeRef = cociService.checkout(fileNodeRef);
 | |
|         // Make sure we can get the checked out node
 | |
|         NodeRef fileWorkingCopyNodeRefCheck = cociService.getWorkingCopy(fileNodeRef);
 | |
|         assertEquals("Working copy not found ", fileWorkingCopyNodeRef, fileWorkingCopyNodeRefCheck);
 | |
|         
 | |
|         // Rename the working copy
 | |
|         nodeService.setProperty(fileWorkingCopyNodeRef, ContentModel.PROP_NAME, "renamed");
 | |
|         
 | |
|         // Check in
 | |
|         cociService.checkin(fileWorkingCopyNodeRef, null);
 | |
|     }
 | |
|     
 | |
|     public void testCheckOutCheckInWithTranslatableAspect()
 | |
|     {
 | |
|         // Create a node to be used as the translation
 | |
|         NodeRef translationNodeRef = nodeService.createNode(
 | |
|                 rootNodeRef,
 | |
|                 ContentModel.ASSOC_CHILDREN,
 | |
|                 QName.createQName("translation"),
 | |
|                 ContentModel.TYPE_CONTENT).getChildRef();
 | |
|         
 | |
|         nodeService.addAspect(this.nodeRef, QName.createQName(NamespaceService.CONTENT_MODEL_1_0_URI, "translatable"), null);
 | |
|         nodeService.createAssociation(this.nodeRef, translationNodeRef, QName.createQName(NamespaceService.CONTENT_MODEL_1_0_URI, "translations"));
 | |
|                 
 | |
|         // Check it out
 | |
|         NodeRef workingCopy = cociService.checkout(
 | |
|                 this.nodeRef, 
 | |
|                 this.rootNodeRef, 
 | |
|                 ContentModel.ASSOC_CHILDREN, 
 | |
|                 QName.createQName("workingCopy"));
 | |
|         
 | |
|                 
 | |
|         // Check it back in again
 | |
|         Map<String, Serializable> versionProperties = new HashMap<String, Serializable>();
 | |
|         versionProperties.put(Version.PROP_DESCRIPTION, "This is a test version");
 | |
|         cociService.checkin(workingCopy, versionProperties);
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Test when the aspect is not set when check-in is performed
 | |
|      */
 | |
|     public void testVersionAspectNotSetOnCheckIn()
 | |
|     {
 | |
|         // Create a bag of props
 | |
|         Map<QName, Serializable> bagOfProps = createTypePropertyBag();
 | |
|         bagOfProps.put(ContentModel.PROP_CONTENT, new ContentData(null, MimetypeMap.MIMETYPE_TEXT_PLAIN, 0L, "UTF-8"));
 | |
| 
 | |
|         // Create a new node 
 | |
|         ChildAssociationRef childAssocRef = nodeService.createNode(
 | |
|                 rootNodeRef,
 | |
|                 ContentModel.ASSOC_CHILDREN,
 | |
|                 QName.createQName("test"),
 | |
|                 ContentModel.TYPE_CONTENT,
 | |
|                 bagOfProps);
 | |
|         NodeRef noVersionNodeRef = childAssocRef.getChildRef();
 | |
|         
 | |
|         // Check out and check in
 | |
|         NodeRef workingCopy = cociService.checkout(noVersionNodeRef);
 | |
|         cociService.checkin(workingCopy, new HashMap<String, Serializable>());
 | |
|         
 | |
|         // Check that the origional node has no version history dispite sending verion props
 | |
|         assertNull(this.versionService.getVersionHistory(noVersionNodeRef));        
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Test cancel checkOut
 | |
|      */
 | |
|     public void testCancelCheckOut()
 | |
|     {
 | |
|         NodeRef workingCopy = checkout();
 | |
|         assertNotNull(workingCopy);
 | |
|         
 | |
|         try
 | |
|         {
 | |
|             this.lockService.checkForLock(this.nodeRef);
 | |
|             fail("The origional should be locked now.");
 | |
|         }
 | |
|         catch (Throwable exception)
 | |
|         {
 | |
|             // Good the origional is locked
 | |
|         }
 | |
|         
 | |
|         NodeRef origNodeRef = cociService.cancelCheckout(workingCopy);
 | |
|         assertEquals(this.nodeRef, origNodeRef);
 | |
|         
 | |
|         // The origional should no longer be locked
 | |
|         this.lockService.checkForLock(origNodeRef);
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Test the deleting a wokring copy node removed the lock on the original node
 | |
|      */
 | |
|     public void testAutoCancelCheckOut()
 | |
|     {
 | |
|         Date modifiedDateBeforeCheckOut = (Date) this.nodeService.getProperty(this.nodeRef, ContentModel.PROP_MODIFIED);
 | |
|         NodeRef workingCopy = checkout();
 | |
|         assertNotNull(workingCopy);
 | |
|         
 | |
|         try
 | |
|         {
 | |
|             this.lockService.checkForLock(this.nodeRef);
 | |
|             fail("The original should be locked now.");
 | |
|         }
 | |
|         catch (Throwable exception)
 | |
|         {
 | |
|             // Good the original is locked
 | |
|         }
 | |
|         
 | |
|         try {Thread.sleep(2000); } catch (InterruptedException e) {}
 | |
|         
 | |
|         // Delete the working copy
 | |
|         nodeService.deleteNode(workingCopy);
 | |
|         
 | |
|         //Make sure that modidied date wasn't changed
 | |
|         Date modifiedDateAfterCheckOut = (Date) this.nodeService.getProperty(this.nodeRef, ContentModel.PROP_MODIFIED);
 | |
|         assertEquals(modifiedDateBeforeCheckOut, modifiedDateAfterCheckOut);
 | |
|         
 | |
|         // The original should no longer be locked
 | |
|         this.lockService.checkForLock(this.nodeRef);
 | |
|         
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * @see CheckOutCheckInService#getWorkingCopy(NodeRef)
 | |
|      * @see CheckOutCheckInService#getCheckedOut(NodeRef)
 | |
|      */
 | |
|     public void testBidirectionalReferences()
 | |
|     {
 | |
|         final NodeRef origNodeRef = nodeService.createNode(
 | |
|                 this.rootNodeRef,
 | |
|                 ContentModel.ASSOC_CHILDREN,
 | |
|                 QName.createQName("test2"),
 | |
|                 ContentModel.TYPE_CONTENT).getChildRef();
 | |
|         
 | |
|         NodeRef wk1 = cociService.getWorkingCopy(origNodeRef);
 | |
|         assertNull(wk1);
 | |
| 
 | |
|         // Check the document out
 | |
|         final NodeRef workingCopy = cociService.checkout(origNodeRef);
 | |
|         assertTrue("Expect cm:workingcopy aspect", nodeService.hasAspect(workingCopy, ContentModel.ASPECT_WORKING_COPY));
 | |
|         assertTrue("Expect cm:checkedOut aspect", nodeService.hasAspect(origNodeRef, ContentModel.ASPECT_CHECKED_OUT));
 | |
|         List<AssociationRef> targetAssocs = nodeService.getTargetAssocs(origNodeRef, ContentModel.ASSOC_WORKING_COPY_LINK);
 | |
|         assertEquals("Expect a 1:1 relationship", 1, targetAssocs.size());
 | |
|         List<AssociationRef> sourceAssocs = nodeService.getSourceAssocs(workingCopy, ContentModel.ASSOC_WORKING_COPY_LINK);
 | |
|         assertEquals("Expect a 1:1 relationship", 1, sourceAssocs.size());
 | |
|         
 | |
|         // Need to commit the transaction in order to get the indexer to run
 | |
|         setComplete();
 | |
|         endTransaction();
 | |
|         
 | |
|         final NodeRef finalNodeRef = origNodeRef;        
 | |
|         
 | |
|         NodeRef wk3 = this.transactionService.getRetryingTransactionHelper().doInTransaction(
 | |
|                 new RetryingTransactionCallback<NodeRef>()
 | |
|                 {
 | |
|                     public NodeRef execute()
 | |
|                     {
 | |
|                         NodeRef wk2 = cociService.getWorkingCopy(finalNodeRef);
 | |
|                         assertNotNull(wk2);
 | |
|                         assertEquals(workingCopy, wk2);
 | |
|                         NodeRef orig2 = cociService.getCheckedOut(wk2);
 | |
|                         assertNotNull(orig2);
 | |
|                         assertEquals(origNodeRef, orig2);
 | |
|                         
 | |
|                         cociService.cancelCheckout(workingCopy);   
 | |
|                         return cociService.getWorkingCopy(nodeRef);
 | |
|                     }
 | |
|                 });
 | |
|         assertNull(wk3);
 | |
|     }
 | |
|     /**
 | |
|      * Test the getWorkingCopy method
 | |
|      */
 | |
|     public void testETWOTWO_733()
 | |
|     {
 | |
|         NodeRef origNodeRef = nodeService.createNode(
 | |
|                 this.rootNodeRef,
 | |
|                 ContentModel.ASSOC_CHILDREN,
 | |
|                 QName.createQName("test2"),
 | |
|                 ContentModel.TYPE_CONTENT).getChildRef();
 | |
|         
 | |
|         // Make a copy of the node
 | |
|         this.copyService.copyAndRename(
 | |
|                 origNodeRef,
 | |
|                 this.rootNodeRef,
 | |
|                 ContentModel.ASSOC_CHILDREN,
 | |
|                 QName.createQName("test6"),
 | |
|                 false);        
 | |
|         
 | |
|         NodeRef wk1 = cociService.getWorkingCopy(origNodeRef);
 | |
|         assertNull(wk1);
 | |
| 
 | |
|         // Check the document out
 | |
|         final NodeRef workingCopy = cociService.checkout(origNodeRef);
 | |
|         
 | |
|         // Need to commit the transaction in order to get the indexer to run
 | |
|         setComplete();
 | |
|         endTransaction();
 | |
|         
 | |
|         final NodeRef finalNodeRef = origNodeRef;        
 | |
|         
 | |
|         NodeRef wk3 = this.transactionService.getRetryingTransactionHelper().doInTransaction(
 | |
|                 new RetryingTransactionCallback<NodeRef>()
 | |
|                 {
 | |
|                     public NodeRef execute()
 | |
|                     {
 | |
|                         NodeRef wk2 = cociService.getWorkingCopy(finalNodeRef);
 | |
|                         assertNotNull(wk2);
 | |
|                         assertEquals(workingCopy, wk2);
 | |
|                         
 | |
|                         cociService.cancelCheckout(workingCopy);                        
 | |
|                         return cociService.getWorkingCopy(nodeRef);
 | |
|                     }
 | |
|                 });
 | |
|         assertNull(wk3);           
 | |
|     }
 | |
|     
 | |
|     public void testAR1056()
 | |
|     {
 | |
|         // Check out the node
 | |
|         NodeRef workingCopy = cociService.checkout(
 | |
|                 this.nodeRef, 
 | |
|                 this.rootNodeRef, 
 | |
|                 ContentModel.ASSOC_CHILDREN, 
 | |
|                 QName.createQName("workingCopy"));
 | |
|         assertNotNull(workingCopy);
 | |
|         
 | |
|         // Try and check the same node out again
 | |
|         try
 | |
|         {
 | |
|             cociService.checkout(
 | |
|                 this.nodeRef, 
 | |
|                 this.rootNodeRef, 
 | |
|                 ContentModel.ASSOC_CHILDREN, 
 | |
|                 QName.createQName("workingCopy2"));
 | |
|             fail("This document has been checked out twice.");
 | |
|         }
 | |
|         catch (Exception exception)
 | |
|         {
 | |
|             // Good because we shouldn't be able to checkout a document twice
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     public void testMultipleCheckoutsCheckInsWithPropChange()
 | |
|     {
 | |
|         // Note: this test assumes cm:autoVersionProps=true by default (refer to cm:versionableAspect in contentModel.xml)
 | |
|         
 | |
|         // Create a new node 
 | |
|         ChildAssociationRef childAssocRef = nodeService.createNode(
 | |
|                 rootNodeRef,
 | |
|                 ContentModel.ASSOC_CHILDREN,
 | |
|                 QName.createQName("test"),
 | |
|                 ContentModel.TYPE_CONTENT,
 | |
|                 null);
 | |
|         final NodeRef testNodeRef = childAssocRef.getChildRef();
 | |
|         
 | |
|         // Add the version aspect to the created node
 | |
|         nodeService.addAspect(testNodeRef, ContentModel.ASPECT_VERSIONABLE, null);
 | |
|         
 | |
|         setComplete();
 | |
|         endTransaction();
 | |
|         
 | |
|         // Checkout
 | |
|         final NodeRef workingCopy1 = transactionService.getRetryingTransactionHelper().doInTransaction(new RetryingTransactionCallback<NodeRef>()
 | |
|         {
 | |
|             public NodeRef execute() throws Exception
 | |
|             {
 | |
|                 return cociService.checkout(testNodeRef);
 | |
|             }
 | |
|         });
 | |
|         
 | |
|         // Change property and checkin
 | |
|         transactionService.getRetryingTransactionHelper().doInTransaction(new RetryingTransactionCallback<Object>()
 | |
|         {
 | |
|             public Object execute() throws Exception
 | |
|             {
 | |
|                 nodeService.setProperty(workingCopy1, ContentModel.PROP_AUTHOR, "author1");
 | |
|                 
 | |
|                 Map<String, Serializable> versionProperties = new HashMap<String, Serializable>();
 | |
|                 versionProperties.put(Version.PROP_DESCRIPTION, "This is a test version 1");
 | |
|                 cociService.checkin(workingCopy1, versionProperties);
 | |
|                 
 | |
|                 return null;
 | |
|             }
 | |
|         });
 | |
|         
 | |
|         // Checkout
 | |
|         final NodeRef workingCopy2 = transactionService.getRetryingTransactionHelper().doInTransaction(new RetryingTransactionCallback<NodeRef>()
 | |
|         {
 | |
|             public NodeRef execute() throws Exception
 | |
|             {
 | |
|                 return cociService.checkout(testNodeRef);
 | |
|             }
 | |
|         });
 | |
|         
 | |
|         // Change property and checkin
 | |
|         transactionService.getRetryingTransactionHelper().doInTransaction(new RetryingTransactionCallback<Object>()
 | |
|         {
 | |
|             public Object execute() throws Exception
 | |
|             {
 | |
|                 nodeService.setProperty(workingCopy2, ContentModel.PROP_AUTHOR, "author2");
 | |
|                 
 | |
|                 Map<String, Serializable> versionProperties = new HashMap<String, Serializable>();
 | |
|                 versionProperties.put(Version.PROP_DESCRIPTION, "This is a test version 2");
 | |
|                 cociService.checkin(workingCopy2, versionProperties);
 | |
|                 
 | |
|                 return null;
 | |
|             }
 | |
|         });
 | |
|         
 | |
|         // Checkout
 | |
|         final NodeRef workingCopy3 = transactionService.getRetryingTransactionHelper().doInTransaction(new RetryingTransactionCallback<NodeRef>()
 | |
|         {
 | |
|             public NodeRef execute() throws Exception
 | |
|             {
 | |
|                 return cociService.checkout(testNodeRef);
 | |
|             }
 | |
|         });
 | |
|         
 | |
|         // Change property and checkin
 | |
|         transactionService.getRetryingTransactionHelper().doInTransaction(new RetryingTransactionCallback<Object>()
 | |
|         {
 | |
|             public Object execute() throws Exception
 | |
|             {
 | |
|                 nodeService.setProperty(workingCopy3, ContentModel.PROP_AUTHOR, "author3");
 | |
|                 
 | |
|                 Map<String, Serializable> versionProperties = new HashMap<String, Serializable>();
 | |
|                 versionProperties.put(Version.PROP_DESCRIPTION, "This is a test version 3");
 | |
|                 cociService.checkin(workingCopy3, versionProperties);
 | |
|                 
 | |
|                 return null;
 | |
|             }
 | |
|         });
 | |
|     }
 | |
|     
 | |
|     public void testAlfrescoCheckoutDoesNotModifyNode()
 | |
|     {
 | |
|         String adminUser = AuthenticationUtil.getAdminUserName();
 | |
|         AuthenticationUtil.setFullyAuthenticatedUser(adminUser);
 | |
|         
 | |
|         Serializable initModifier = nodeService.getProperty(nodeRef, ContentModel.PROP_MODIFIER);
 | |
|         Serializable initModified = nodeService.getProperty(nodeRef, ContentModel.PROP_MODIFIED);
 | |
|         assertFalse("The initial modifier should not be Admin!", adminUser.equals(initModifier));
 | |
|         
 | |
|         NodeRef copy = cociService.checkout(
 | |
|                 nodeRef, 
 | |
|                 rootNodeRef, 
 | |
|                 ContentModel.ASSOC_CHILDREN, 
 | |
|                 QName.createQName("workingCopy"));
 | |
| 
 | |
|         Serializable modifier = nodeService.getProperty(nodeRef, ContentModel.PROP_MODIFIER);
 | |
|         assertEquals("Checkout should not cause the modifier to change!", initModifier, modifier);
 | |
|         Serializable modified = nodeService.getProperty(nodeRef, ContentModel.PROP_MODIFIED);
 | |
|         assertEquals("Checkout should not cause the modified date to change!", initModified, modified);
 | |
| 
 | |
|         cociService.cancelCheckout(copy);
 | |
|         modifier = nodeService.getProperty(nodeRef, ContentModel.PROP_MODIFIER);
 | |
|         assertEquals("Cancel checkout should not cause the modifier to change!", initModifier, modifier);
 | |
|         modified = nodeService.getProperty(nodeRef, ContentModel.PROP_MODIFIED);
 | |
|         assertEquals("Cancel checkout should not cause the modified date to change!", initModified, modified);
 | |
|         
 | |
|         copy = cociService.checkout(
 | |
|                 nodeRef, 
 | |
|                 rootNodeRef, 
 | |
|                 ContentModel.ASSOC_CHILDREN, 
 | |
|                 QName.createQName("workingCopy"));
 | |
|         Map<String, Serializable> versionProperties = new HashMap<String, Serializable>();
 | |
|         versionProperties.put(Version.PROP_DESCRIPTION, "This is a test version");      
 | |
|         cociService.checkin(copy, versionProperties);
 | |
|         
 | |
|         modifier = nodeService.getProperty(nodeRef, ContentModel.PROP_MODIFIER);
 | |
|         // See MNT-8789, nothing has changed in the working copy, so the modifier should be left untouched
 | |
|         assertEquals("The modifier should NOT change to Admin after checkin!", initModifier, modifier);
 | |
|     }
 | |
|     
 | |
|     public void testCheckOutPermissions_ALF7680_ALF535()
 | |
|     {
 | |
|         /*
 | |
|          * Testing working copy creation in folder of source node. 
 | |
|          * User has no permissions to create children in this folder. 
 | |
|          * User has permissions to edit document.
 | |
|          * Expected result: working copy should be created.
 | |
|          */
 | |
|         
 | |
|         NodeRef folder1 = createFolderWithPermission(rootNodeRef, userName, PermissionService.CONSUMER);
 | |
|         NodeRef node = createNodeWithPermission(folder1, userName, PermissionService.EDITOR);
 | |
| 
 | |
|         // Check out the node
 | |
|         NodeRef workingCopy = cociService.checkout(
 | |
|                 node, 
 | |
|                 folder1, 
 | |
|                 ContentModel.ASSOC_CHILDREN, 
 | |
|                 QName.createQName("workingCopy"));
 | |
| 
 | |
|         // Ensure that the working copy was created and current user was set as owner
 | |
|         assertNotNull(workingCopy);
 | |
|         assertTrue(nodeService.hasAspect(workingCopy, ContentModel.ASPECT_WORKING_COPY));
 | |
|         assertEquals(this.userNodeRef, nodeService.getProperty(workingCopy, ContentModel.PROP_WORKING_COPY_OWNER));
 | |
| 
 | |
|         cociService.cancelCheckout(workingCopy);
 | |
| 
 | |
|         /*
 | |
|          * Testing working copy creation in a different folder. 
 | |
|          * User has permissions to create children in this folder. 
 | |
|          * User has permissions to edit document.
 | |
|          * Expected result: working copy should be created.
 | |
|          */
 | |
|         
 | |
|         NodeRef folder2 = createFolderWithPermission(rootNodeRef, userName, PermissionService.ALL_PERMISSIONS);
 | |
| 
 | |
|         // Check out the node
 | |
|         workingCopy = cociService.checkout(
 | |
|                 node, 
 | |
|                 folder2, 
 | |
|                 ContentModel.ASSOC_CHILDREN, 
 | |
|                 QName.createQName("workingCopy"));
 | |
| 
 | |
|         // Ensure that the working copy was created and current user was set as owner
 | |
|         assertNotNull(workingCopy);
 | |
|         assertTrue(nodeService.hasAspect(workingCopy, ContentModel.ASPECT_WORKING_COPY));
 | |
|         assertEquals(this.userNodeRef, nodeService.getProperty(workingCopy, ContentModel.PROP_WORKING_COPY_OWNER));
 | |
| 
 | |
|         cociService.cancelCheckout(workingCopy);
 | |
| 
 | |
|         /*
 | |
|          * Testing working copy creation in a different folder. 
 | |
|          * User has no permissions to create children in this folder. 
 | |
|          * User has permissions to edit document.
 | |
|          * Expected result: exception.
 | |
|          */
 | |
|         
 | |
|         NodeRef folder3 = createFolderWithPermission(rootNodeRef, userName, PermissionService.CONSUMER);
 | |
|         try
 | |
|         {
 | |
|             // Check out the node
 | |
|             workingCopy = cociService.checkout(
 | |
|                     node, 
 | |
|                     folder3, 
 | |
|                     ContentModel.ASSOC_CHILDREN, 
 | |
|                     QName.createQName("workingCopy"));
 | |
| 
 | |
|             // Ensure that the working copy was not created and exception occurs
 | |
|             fail("Node can't be checked out to folder where user has no permissions to create children");
 | |
|         }
 | |
|         catch (Exception e)
 | |
|         {
 | |
|             // Exception is expected
 | |
|         }
 | |
|         
 | |
|         /*
 | |
|          * Testing working copy creation in a different folder. 
 | |
|          * User has permissions to create children in this folder. 
 | |
|          * User has no permissions to edit document. 
 | |
|          * Expected result: exception.
 | |
|          */
 | |
|         
 | |
|         NodeRef node2 = createNodeWithPermission(folder3, userName, PermissionService.CONSUMER);
 | |
|         try
 | |
|         {
 | |
|             // Check out the node
 | |
|             workingCopy = cociService.checkout(
 | |
|                     node2, 
 | |
|                     folder3, 
 | |
|                     ContentModel.ASSOC_CHILDREN, 
 | |
|                     QName.createQName("workingCopy"));
 | |
| 
 | |
|             // Ensure that the working copy was not created and exception occurs
 | |
|             fail("Node can't be checked out if user has no permissions to edit document");
 | |
|         }
 | |
|         catch (Exception e)
 | |
|         {
 | |
|             // Exception is expected
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public void testCheckInLockableAspectDoesntCopies_ALF16194()
 | |
|     {
 | |
|         // Check-out nodeRef
 | |
|         NodeRef workingCopy = this.cociService.checkout(
 | |
|                 this.nodeRef, 
 | |
|                 this.rootNodeRef, 
 | |
|                 ContentModel.ASSOC_CHILDREN, 
 | |
|                 QName.createQName("workingCopy"));
 | |
|         assertNotNull(workingCopy);
 | |
|         
 | |
|         // Check-in 
 | |
|         Map<String, Serializable> versionProperties = new HashMap<String, Serializable>();
 | |
|         versionProperties.put(Version.PROP_DESCRIPTION, "This is a test version");      
 | |
|         cociService.checkin(workingCopy, versionProperties);
 | |
|         
 | |
|         if(nodeService.hasAspect(nodeRef, ContentModel.ASPECT_LOCKABLE))
 | |
|         {
 | |
|             fail("Lockable aspect should not be copied from the working copy to the original document");
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     private NodeRef createFolderWithPermission(NodeRef parent, String username, String permission)
 | |
|     {
 | |
|         // Authenticate as system user because the current user should not be node owner
 | |
|         AuthenticationComponent authenticationComponent = (AuthenticationComponent) this.applicationContext.getBean("authenticationComponent");
 | |
|         authenticationComponent.setSystemUserAsCurrentUser();
 | |
| 
 | |
|         // Create the folder
 | |
|         NodeRef folder = nodeService.createNode(
 | |
|                 parent, 
 | |
|                 ContentModel.ASSOC_CHILDREN, 
 | |
|                 QName.createQName("TestFolder" + GUID.generate()), 
 | |
|                 ContentModel.TYPE_CONTENT).getChildRef();
 | |
| 
 | |
|         // Apply permissions to folder
 | |
|         permissionService.deletePermissions(folder);
 | |
|         permissionService.setInheritParentPermissions(folder, false);
 | |
|         permissionService.setPermission(folder, userName, permission, true);
 | |
| 
 | |
|         // Authenticate test user
 | |
|         TestWithUserUtils.authenticateUser(this.userName, PWD, this.rootNodeRef, this.authenticationService);
 | |
| 
 | |
|         return folder;
 | |
|     }
 | |
| 
 | |
|     private NodeRef createNodeWithPermission(NodeRef parent, String username, String permission)
 | |
|     {
 | |
|         // Authenticate as system user because the current user should not be node owner
 | |
|         AuthenticationComponent authenticationComponent = (AuthenticationComponent) this.applicationContext.getBean("authenticationComponent");
 | |
|         authenticationComponent.setSystemUserAsCurrentUser();
 | |
| 
 | |
|         // Create the node as a copy of prepared
 | |
|         NodeRef node = copyService.copy(nodeRef, parent, ContentModel.ASSOC_CHILDREN, ContentModel.TYPE_CONTENT);
 | |
| 
 | |
|         // Apply permissions to node
 | |
|         permissionService.deletePermissions(node);
 | |
|         permissionService.setInheritParentPermissions(node, false);
 | |
|         permissionService.setPermission(node, userName, permission, true);
 | |
| 
 | |
|         // Authenticate test user
 | |
|         TestWithUserUtils.authenticateUser(this.userName, PWD, this.rootNodeRef, this.authenticationService);
 | |
| 
 | |
|         return node;
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * MNT-2641
 | |
|      * <p/> 
 | |
|      * Creating a document and working copy. Then try to move working copy to another place. Test is passed, if a working copy was moved to another place with original
 | |
|      * document. Only the lock owner can move documents.
 | |
|      */
 | |
|     public void testMoveOriginalWithWorkingCopy()
 | |
|     {
 | |
|         // Create a FolderA
 | |
|         final NodeRef folderA = createFolder("MoveOriginalWithWorkingCopy_" + GUID.generate());
 | |
| 
 | |
|         // Create a FolderB
 | |
|         final NodeRef folderB = createFolder("MoveOriginalWithWorkingCopy_" + GUID.generate());
 | |
| 
 | |
|         // Create content in FolderA, that allowed to move for current user
 | |
|         NodeRef origAllowed = createContent("original_" + GUID.generate(), folderA);
 | |
| 
 | |
|         // Check out the document, that allowed to move for current user
 | |
|         NodeRef workingCopyAllowed = this.cociService.checkout(origAllowed);
 | |
|         assertNotNull(workingCopyAllowed);
 | |
|         
 | |
|         // Create content in FolderA, that doesn't allowed to move for other users
 | |
|         final NodeRef origDenied = createContent("original_" + GUID.generate(), folderA);
 | |
| 
 | |
|         // Check out the document, that doesn't allowed to move for other users
 | |
|         final NodeRef workingCopyDenied = this.cociService.checkout(origDenied);
 | |
|         assertNotNull(workingCopyDenied);
 | |
| 
 | |
|         // Move working copy to folderB
 | |
|         NodeRef movedWorkingCopyAllowed = null;
 | |
|         try
 | |
|         {
 | |
|             movedWorkingCopyAllowed = fileFolderService.moveFrom(workingCopyAllowed, folderA, folderB, null).getNodeRef();
 | |
|         }
 | |
|         catch (Exception e)
 | |
|         {
 | |
|             // do nothing. Assert condition checks it further
 | |
|         }
 | |
| 
 | |
|         assertNotNull(movedWorkingCopyAllowed);
 | |
| 
 | |
|         // check a parent of moved working copy - it must be folderB
 | |
|         assertEquals(folderB, nodeService.getPrimaryParent(movedWorkingCopyAllowed).getParentRef());
 | |
|         
 | |
|         boolean thrown = false;
 | |
|         
 | |
|         // check a parent of original document - it must be folderA
 | |
|         assertEquals(folderA, nodeService.getPrimaryParent(origAllowed).getParentRef());
 | |
| 
 | |
|         // try to move original - action must be denied
 | |
|         try
 | |
|         {
 | |
|             fileFolderService.moveFrom(origAllowed, folderA, folderB, null).getNodeRef();
 | |
|         }
 | |
|         catch (NodeLockedException e1)
 | |
|         {
 | |
|             thrown = true;
 | |
|         }
 | |
|         catch (Exception e)
 | |
|         {
 | |
|          // do nothing. Assert condition checks it further
 | |
|         }
 | |
|         assertTrue(thrown);
 | |
|         
 | |
|         ////////////////////////////////////////////////
 | |
|         ////////////////////////////////////////////////
 | |
| 
 | |
|         // create another person
 | |
|         final String denyUser = "COCITestUser123";
 | |
|         createPerson(denyUser);
 | |
| 
 | |
|         // try to move a working copy. User hasn't permissions to move
 | |
|         thrown = false;
 | |
|         try
 | |
|         {
 | |
|             AuthenticationUtil.runAs(new RunAsWork<NodeRef>()
 | |
|             {
 | |
|                 @Override
 | |
|                 public NodeRef doWork() throws Exception
 | |
|                 {
 | |
|                     return fileFolderService.moveFrom(workingCopyDenied, folderA, folderB, null).getNodeRef();
 | |
|                 }
 | |
|             }, denyUser);
 | |
|         }
 | |
|         catch (AccessDeniedException e)
 | |
|         {
 | |
|             thrown = true;
 | |
|         }
 | |
|         catch (org.alfresco.repo.security.permissions.AccessDeniedException e)
 | |
|         {
 | |
|             thrown = true;
 | |
|         }
 | |
|         assertTrue(thrown);
 | |
|         
 | |
|         //try to move a original file. User hasn't permissions to move
 | |
|         thrown = false;
 | |
|         try
 | |
|         {
 | |
|             AuthenticationUtil.runAs(new RunAsWork<NodeRef>()
 | |
|             {
 | |
|                 @Override
 | |
|                 public NodeRef doWork() throws Exception
 | |
|                 {
 | |
|                     return fileFolderService.moveFrom(origDenied, folderA, folderB, null).getNodeRef();
 | |
|                 }
 | |
|             }, denyUser);
 | |
|         }
 | |
|         catch (AccessDeniedException e)
 | |
|         {
 | |
|             thrown = true;
 | |
|         }
 | |
|         catch (org.alfresco.repo.security.permissions.AccessDeniedException e)
 | |
|         {
 | |
|             thrown = true;
 | |
|         }
 | |
|         assertTrue(thrown);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * MNT-2641 
 | |
|      */
 | |
|     public void testDeleteUpdateOriginalOfCheckedOutDocument()
 | |
|     {
 | |
|         // Create a FolderA
 | |
|         final NodeRef folderA = createFolder("DeleteUpdateOriginalOfCheckedOutDocument_" + GUID.generate());
 | |
| 
 | |
|         // Create content in FolderA
 | |
|         final NodeRef orig = createContent("original_" + GUID.generate(), folderA);
 | |
| 
 | |
|         // Check out the document
 | |
|         NodeRef workingCopy = this.cociService.checkout(orig);
 | |
|         assertNotNull(workingCopy);
 | |
|         
 | |
|         boolean thrown = false;
 | |
|         
 | |
|         // try to delete original, that has working copy - must be denied
 | |
|         try
 | |
|         {
 | |
|             fileFolderService.delete(orig);
 | |
|         }
 | |
|         catch (NodeLockedException e)
 | |
|         {
 | |
|             thrown = true;
 | |
|         }
 | |
|         assertTrue("No one should be able to delete the original", thrown);
 | |
|         
 | |
|         // creating a properties
 | |
|         final Map<QName, Serializable> propsToPersist = new HashMap<QName, Serializable>(3);
 | |
|         MLText value = new MLText(Locale.ENGLISH, GUID.generate() + "");
 | |
|         propsToPersist.put(ContentModel.PROP_DESCRIPTION, value);
 | |
|         value = new MLText(Locale.ENGLISH, null);
 | |
|         propsToPersist.put(ContentModel.PROP_TITLE, value);
 | |
|         
 | |
|         // try to modify properties of original, that has working copy - must be denied
 | |
|         thrown = false;
 | |
|         try
 | |
|         {
 | |
|             nodeService.addProperties(orig, propsToPersist);
 | |
|         }
 | |
|         catch (NodeLockedException e)
 | |
|         {
 | |
|             thrown = true;
 | |
|         }
 | |
|         assertTrue("No one should be able to update the original", thrown);
 | |
|         
 | |
|         // ////////////////////////////////////////////////////////////
 | |
|         // testing "delete" and "update" actions with non-owner user //
 | |
|         // ////////////////////////////////////////////////////////////
 | |
|         
 | |
|         // create another person
 | |
|         final String denyUser = "COCITestUser123";
 | |
|         createPerson(denyUser);
 | |
| 
 | |
|         // try to delete original, that has working copy - must be denied
 | |
|         thrown = false;
 | |
|         try
 | |
|         {
 | |
|             AuthenticationUtil.runAs(new RunAsWork<NodeRef>()
 | |
|             {
 | |
|                 @Override
 | |
|                 public NodeRef doWork() throws Exception
 | |
|                 {
 | |
|                     fileFolderService.delete(orig);
 | |
|                     return null;
 | |
|                 }
 | |
|             }, denyUser);
 | |
|         }
 | |
|         catch (AccessDeniedException e)
 | |
|         {
 | |
|             thrown = true;
 | |
|         }
 | |
|         catch (org.alfresco.repo.security.permissions.AccessDeniedException e)
 | |
|         {
 | |
|             thrown = true;
 | |
|         }
 | |
|         assertTrue(thrown);
 | |
| 
 | |
|         // try to delete original, that has working copy - must be denied
 | |
|         thrown = false;
 | |
|         try
 | |
|         {
 | |
|             AuthenticationUtil.runAs(new RunAsWork<Void>()
 | |
|             {
 | |
|                 @Override
 | |
|                 public Void doWork() throws Exception
 | |
|                 {
 | |
|                     nodeService.addProperties(orig, propsToPersist);
 | |
|                     return null;
 | |
|                 }
 | |
|             }, denyUser);
 | |
|         }
 | |
|         catch (AccessDeniedException e)
 | |
|         {
 | |
|             thrown = true;
 | |
|         }
 | |
|         catch (org.alfresco.repo.security.permissions.AccessDeniedException e)
 | |
|         {
 | |
|             thrown = true;
 | |
|         }
 | |
|         catch (NodeLockedException e) {
 | |
|             thrown = true;
 | |
|         }
 | |
|         assertTrue(thrown);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * MNT-2641 
 | |
|      * The working copy delete is equivalent to "cancelCheckout". This should fail for everyone except the lock owner.
 | |
|      */
 | |
|     public void testDeleteOfWorkingCopy()
 | |
|     {
 | |
|         // Create a FolderA
 | |
|         final NodeRef folderA = createFolder("DeleteOfWorkingCopy_" + GUID.generate());
 | |
| 
 | |
|         // Create content in FolderA
 | |
|         final NodeRef orig = createContent("original_" + GUID.generate(), folderA);
 | |
| 
 | |
|         // Check out the document
 | |
|         NodeRef workingCopy = this.cociService.checkout(orig);
 | |
|         assertNotNull(workingCopy);
 | |
|         
 | |
|         // deleting of working copy
 | |
|         fileFolderService.delete(workingCopy);
 | |
|         assertFalse(nodeService.exists(workingCopy));
 | |
|         
 | |
|         assertNull(cociService.getWorkingCopy(orig));
 | |
|         assertFalse(cociService.isCheckedOut(orig));
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * MNT-2641: The {@link ContentModel#ASPECT_WORKING_COPY} aspect cannot be removed from a working copy
 | |
|      */
 | |
|     public void testDeleteWorkingCopyAspect()
 | |
|     {
 | |
|         // Create a FolderA
 | |
|         final NodeRef folderA = createFolder("DeleteCopiedFromAspectFromWorkingCopy_" + GUID.generate());
 | |
| 
 | |
|         // Create content in FolderA
 | |
|         final NodeRef orig = createContent("original_" + GUID.generate(), folderA);
 | |
| 
 | |
|         // Check out the document
 | |
|         NodeRef workingCopy = this.cociService.checkout(orig);
 | |
|         assertNotNull(workingCopy);
 | |
|         
 | |
|         assertTrue("cm:workingCopy aspect not found on working copy.",
 | |
|                 nodeService.hasAspect(workingCopy, ContentModel.ASPECT_WORKING_COPY));
 | |
|         assertTrue("cm:copiedFrom aspect not found on working copy.",
 | |
|                 nodeService.hasAspect(workingCopy, ContentModel.ASPECT_COPIEDFROM));
 | |
|         
 | |
|         setComplete();
 | |
|         endTransaction();
 | |
|         
 | |
|         // try to delete cm:copiedfrom aspect from working copy - must be allowed
 | |
|         nodeService.removeAspect(workingCopy, ContentModel.ASPECT_COPIEDFROM);
 | |
|         // Try to delete cm:workingcopy aspect from working copy - must be denied
 | |
|         try
 | |
|         {
 | |
|             nodeService.removeAspect(workingCopy, ContentModel.ASPECT_WORKING_COPY);
 | |
|             fail("Should not be able to remove cm:workingcopy");
 | |
|         }
 | |
|         catch (UnsupportedOperationException e)
 | |
|         {
 | |
|             // Expected
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * MNT-2641 The cm:workingcopylink association cannot be removed
 | |
|      */
 | |
|     public void testDeleteWorkingCopyLinkAssociation()
 | |
|     {
 | |
|      // Create a FolderA
 | |
|         final NodeRef folderA = createFolder("DeleteOriginalAssociationFromCopy_" + GUID.generate());
 | |
| 
 | |
|         // Create content in FolderA
 | |
|         final NodeRef orig = createContent("original_" + GUID.generate(), folderA);
 | |
| 
 | |
|         // Check out the document
 | |
|         NodeRef workingCopy = this.cociService.checkout(orig);
 | |
|         assertNotNull(workingCopy);
 | |
|         
 | |
|         // Check that the cm:original association is present
 | |
|         assertEquals("Did not find cm:workingcopylink",
 | |
|                 1, nodeService.getSourceAssocs(workingCopy, ContentModel.ASSOC_WORKING_COPY_LINK).size());
 | |
|         
 | |
|         setComplete();
 | |
|         endTransaction();
 | |
|         
 | |
|         // try to delete cm:workingcopylink association - must be denied
 | |
|         try
 | |
|         {
 | |
|             nodeService.removeAssociation(orig, workingCopy, ContentModel.ASSOC_WORKING_COPY_LINK);
 | |
|             fail("Should not have been allowed to remove the association from working copy to original");
 | |
|         }
 | |
|         catch (IntegrityException e)
 | |
|         {
 | |
|             // Expected
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     private NodeRef createFolder(String fName)
 | |
|     {
 | |
|         return nodeService.createNode(rootNodeRef, ContentModel.ASSOC_CHILDREN, QName.createQName(fName), ContentModel.TYPE_FOLDER).getChildRef();
 | |
|     }
 | |
| 
 | |
|     private NodeRef createContent(String contentName, NodeRef parentRef)
 | |
|     {
 | |
|         return nodeService.createNode(parentRef, ContentModel.ASSOC_CONTAINS, QName.createQName(contentName), ContentModel.TYPE_CONTENT).getChildRef();
 | |
|     }
 | |
|     private void createPerson(String userName)
 | |
|     {
 | |
|         // if user with given user name doesn't already exist then create user
 | |
|         if (this.authenticationService.authenticationExists(userName) == false)
 | |
|         {
 | |
|             // create user
 | |
|             this.authenticationService.createAuthentication(userName, "password".toCharArray());
 | |
|         }
 | |
| 
 | |
|         // if person node with given user name doesn't already exist then create
 | |
|         // person
 | |
|         if (this.personService.personExists(userName) == false)
 | |
|         {
 | |
|             // create person properties
 | |
|             final PropertyMap personProps = new PropertyMap();
 | |
|             personProps.put(ContentModel.PROP_USERNAME, userName);
 | |
|             personProps.put(ContentModel.PROP_FIRSTNAME, userName);
 | |
|             personProps.put(ContentModel.PROP_LASTNAME, userName);
 | |
|             personProps.put(ContentModel.PROP_EMAIL, userName + "@gmail.com");
 | |
|             personProps.put(ContentModel.PROP_JOBTITLE, "jobtitle");
 | |
|             personProps.put(ContentModel.PROP_ORGANIZATION, "org");
 | |
| 
 | |
|             // create person node for user
 | |
|             AuthenticationUtil.runAsSystem(new RunAsWork<NodeRef>()
 | |
|             {
 | |
|                 @Override
 | |
|                 public NodeRef doWork() throws Exception
 | |
|                 {
 | |
|                     return personService.createPerson(personProps);
 | |
|                 }
 | |
|             });
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * <br>
 | |
|      * Creating node - CheckOut - Add write lock to working copy - Unlock working copy - CancelCheckOut
 | |
|      */
 | |
|     public void testCancelCheckoutUnlockedWCopy()
 | |
|     {
 | |
|         ServiceRegistry serviceRegistry = (ServiceRegistry)this.applicationContext.getBean(ServiceRegistry.SERVICE_REGISTRY);
 | |
|         CheckOutCheckInService securityCOCIService = serviceRegistry.getCheckOutCheckInService();
 | |
|         NodeRef folderA = createFolder(rootNodeRef, "testMnt9502_" + System.currentTimeMillis());
 | |
|         assertNotNull(folderA);
 | |
|         NodeRef clucc = createContent("checkout_lock_unlock_cancelCO", folderA);
 | |
|         assertNotNull(clucc);
 | |
|         
 | |
|         NodeRef wc = securityCOCIService.checkout(clucc);
 | |
|         lockService.lock(wc, LockType.WRITE_LOCK, 60*60);
 | |
|         lockService.unlock(wc);
 | |
|         securityCOCIService.cancelCheckout(wc);
 | |
|     }
 | |
|     
 | |
|     private NodeRef createFolder(NodeRef rootNodeRef, String fName)
 | |
|     {
 | |
|         return nodeService.createNode(rootNodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName(fName), ContentModel.TYPE_FOLDER).getChildRef();
 | |
|     }
 | |
| }
 |