/* * Copyright (C) 2009-2010 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 . */ package org.alfresco.repo.transfer; import static org.mockito.Mockito.mock; import static org.mockito.Mockito.when; import java.util.ArrayList; import java.util.Collection; import java.util.Date; import java.util.HashSet; import java.util.List; import java.util.Locale; import java.util.Queue; import java.util.Set; import javax.transaction.UserTransaction; import javax.xml.XMLConstants; import javax.xml.transform.Source; import javax.xml.transform.stream.StreamSource; import javax.xml.validation.Schema; import javax.xml.validation.SchemaFactory; import javax.xml.validation.Validator; import org.alfresco.model.ContentModel; import org.alfresco.repo.security.authentication.AuthenticationComponent; import org.alfresco.repo.security.authentication.AuthenticationUtil; import org.alfresco.repo.transfer.manifest.TransferManifestNodeFactory; import org.alfresco.service.cmr.action.ActionService; import org.alfresco.service.cmr.lock.LockService; import org.alfresco.service.cmr.lock.LockType; import org.alfresco.service.cmr.repository.ChildAssociationRef; 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.NodeRef; import org.alfresco.service.cmr.repository.NodeService; import org.alfresco.service.cmr.repository.Path; import org.alfresco.service.cmr.repository.StoreRef; import org.alfresco.service.cmr.search.ResultSet; import org.alfresco.service.cmr.search.SearchService; import org.alfresco.service.cmr.security.AccessPermission; 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.transfer.TransferCallback; import org.alfresco.service.cmr.transfer.TransferDefinition; import org.alfresco.service.cmr.transfer.TransferEvent; import org.alfresco.service.cmr.transfer.TransferEventBegin; import org.alfresco.service.cmr.transfer.TransferException; import org.alfresco.service.cmr.transfer.TransferReceiver; import org.alfresco.service.cmr.transfer.TransferService; import org.alfresco.service.cmr.transfer.TransferTarget; import org.alfresco.service.descriptor.Descriptor; import org.alfresco.service.descriptor.DescriptorService; import org.alfresco.service.namespace.NamespaceService; import org.alfresco.service.namespace.QName; import org.alfresco.service.transaction.TransactionService; import org.alfresco.util.BaseAlfrescoSpringTest; import org.alfresco.util.GUID; import org.alfresco.util.Pair; import org.alfresco.util.PropertyMap; import org.springframework.transaction.TransactionDefinition; import org.springframework.transaction.support.DefaultTransactionDefinition; import org.springframework.util.ResourceUtils; /** * Unit test for TransferServiceImpl * * Contains some integration tests for the transfer service * * @author Mark Rogers */ @SuppressWarnings("deprecation") public class TransferServiceImplTest extends BaseAlfrescoSpringTest { private TransferService transferService; private ContentService contentService; private TransferServiceImpl transferServiceImpl; private SearchService searchService; private TransactionService transactionService; private TransferReceiver receiver; private TransferManifestNodeFactory transferManifestNodeFactory; private PermissionService permissionService; private LockService lockService; private PersonService personService; private DescriptorService descriptorService; String COMPANY_HOME_XPATH_QUERY = "/{http://www.alfresco.org/model/application/1.0}company_home"; String GUEST_HOME_XPATH_QUERY = "/{http://www.alfresco.org/model/application/1.0}company_home/{http://www.alfresco.org/model/application/1.0}guest_home"; String REPO_ID_A = "RepoIdA"; String REPO_ID_B = "RepoIdB"; String REPO_ID_C = "RepoIdC"; @Override public void runBare() throws Throwable { preventTransaction(); super.runBare(); } /** * Called during the transaction setup */ protected void onSetUp() throws Exception { super.onSetUp(); // Get the required services this.transferService = (TransferService)this.applicationContext.getBean("TransferService"); this.contentService = (ContentService)this.applicationContext.getBean("ContentService"); this.transferServiceImpl = (TransferServiceImpl)this.applicationContext.getBean("transferService"); this.searchService = (SearchService)this.applicationContext.getBean("SearchService"); this.transactionService = (TransactionService)this.applicationContext.getBean("TransactionService"); this.nodeService = (NodeService) this.applicationContext.getBean("nodeService"); this.contentService = (ContentService) this.applicationContext.getBean("contentService"); this.authenticationService = (MutableAuthenticationService) this.applicationContext.getBean("authenticationService"); this.actionService = (ActionService)this.applicationContext.getBean("actionService"); this.permissionService = (PermissionService)this.applicationContext.getBean("permissionService"); this.receiver = (TransferReceiver)this.applicationContext.getBean("transferReceiver"); this.transferManifestNodeFactory = (TransferManifestNodeFactory)this.applicationContext.getBean("transferManifestNodeFactory"); this.authenticationComponent = (AuthenticationComponent) this.applicationContext.getBean("authenticationComponent"); this.lockService = (LockService) this.applicationContext.getBean("lockService"); this.personService = (PersonService)this.applicationContext.getBean("PersonService"); this.descriptorService = (DescriptorService)this.applicationContext.getBean("DescriptorService"); authenticationComponent.setSystemUserAsCurrentUser(); setTransactionDefinition(new DefaultTransactionDefinition(TransactionDefinition.PROPAGATION_REQUIRES_NEW)); assertNotNull("receiver is null", this.receiver); } /** * Test create target. * * @throws Exception */ public void testCreateTarget() throws Exception { String name = "name"; String title = "title"; String description = "description"; String endpointProtocol = "http"; String endpointHost = "localhost"; int endpointPort = 8080; String endpointPath = "rhubarb"; String username = "admin"; char[] password = "password".toCharArray(); startNewTransaction(); try { /** * Now go ahead and create our first transfer target */ TransferTarget ret = transferService.createAndSaveTransferTarget(name, title, description, endpointProtocol, endpointHost, endpointPort, endpointPath, username, password); assertNotNull("return value is null", ret); assertNotNull("node ref is null", ret.getNodeRef()); //titled aspect assertEquals("name not equal", ret.getName(), name); assertEquals("title not equal", ret.getTitle(), title); assertEquals("description not equal", ret.getDescription(), description); // endpoint assertEquals("endpointProtocol not equal", ret.getEndpointProtocol(), endpointProtocol); assertEquals("endpointHost not equal", ret.getEndpointHost(), endpointHost); assertEquals("endpointPort not equal", ret.getEndpointPort(), endpointPort); assertEquals("endpointPath not equal", ret.getEndpointPath(), endpointPath); // authentication assertEquals("username not equal", ret.getUsername(), username); char[] password2 = ret.getPassword(); assertEquals(password.length, password2.length); for(int i = 0; i < password.length; i++) { if(password[i] != password2[i]) { fail("password not equal:" + new String(password) + new String(password2)); } } /** * Negative test - try to create a transfer target with a name that's already used. */ transferService.createAndSaveTransferTarget(name, title, description, endpointProtocol, endpointHost, endpointPort, endpointPath, username, password); fail("duplicate name not detected"); } catch (TransferException e) { // expect to go here } finally { endTransaction(); } } /** * Test create target via in memory data object. * * @throws Exception */ public void testCreateTargetSyntax2() throws Exception { String name = "nameSyntax2"; String title = "title"; String description = "description"; String endpointProtocol = "http"; String endpointHost = "localhost"; int endpointPort = 8080; String endpointPath = "rhubarb"; String username = "admin"; char[] password = "password".toCharArray(); startNewTransaction(); try { /** * Now go ahead and create our first transfer target */ TransferTarget newValue = transferService.createTransferTarget(name); newValue.setDescription(description); newValue.setEndpointHost(endpointHost); newValue.setEndpointPort(endpointPort); newValue.setEndpointPath(endpointPath); newValue.setEndpointProtocol(endpointProtocol); newValue.setPassword(password); newValue.setTitle(title); newValue.setUsername(username); TransferTarget ret = transferService.saveTransferTarget(newValue); assertNotNull("return value is null", ret); assertNotNull("node ref is null", ret.getNodeRef()); //titled aspect assertEquals("name not equal", ret.getName(), name); assertEquals("title not equal", ret.getTitle(), title); assertEquals("description not equal", ret.getDescription(), description); // endpoint assertEquals("endpointProtocol not equal", ret.getEndpointProtocol(), endpointProtocol); assertEquals("endpointHost not equal", ret.getEndpointHost(), endpointHost); assertEquals("endpointPort not equal", ret.getEndpointPort(), endpointPort); assertEquals("endpointPath not equal", ret.getEndpointPath(), endpointPath); // authentication assertEquals("username not equal", ret.getUsername(), username); char[] password2 = ret.getPassword(); assertEquals(password.length, password2.length); for(int i = 0; i < password.length; i++) { if(password[i] != password2[i]) { fail("password not equal:" + new String(password) + new String(password2)); } } /** * Negative test - try to create a transfer target with a name that's already used. */ transferService.createAndSaveTransferTarget(name, title, description, endpointProtocol, endpointHost, endpointPort, endpointPath, username, password); fail("duplicate name not detected"); } catch (TransferException e) { // expect to go here } finally { endTransaction(); } } /** * Test of Get TransferTargets * * @throws Exception */ public void testGetTransferTargets() throws Exception { String nameA = "nameA"; String nameB = "nameB"; String title = "title"; String description = "description"; String endpointProtocol = "http"; String endpointHost = "localhost"; int endpointPort = 8080; String endpointPath = "rhubarb"; String username = "admin"; char[] password = "password".toCharArray(); startNewTransaction(); try { /** * Now go ahead and create our first transfer target */ TransferTarget targetA = transferService.createAndSaveTransferTarget(nameA, title, description, endpointProtocol, endpointHost, endpointPort, endpointPath, username, password); TransferTarget targetB = transferService.createAndSaveTransferTarget(nameB, title, description, endpointProtocol, endpointHost, endpointPort, endpointPath, username, password); Set targets = transferService.getTransferTargets(); assertTrue("targets is empty", targets.size() > 0); assertTrue("didn't find target A", targets.contains(targetA) ); assertTrue("didn't find target B", targets.contains(targetB)); for(TransferTarget target : targets) { System.out.println("found target: " + target.getName()); } } finally { endTransaction(); } } /** * Test of Get All Transfer Targets By Group */ //TODO Test not complete - can't yet put targets in different groups public void testGetAllTransferTargetsByGroup() throws Exception { String getMe = "getMe"; String title = "title"; String description = "description"; String endpointProtocol = "http"; String endpointHost = "localhost"; int endpointPort = 8080; String endpointPath = "rhubarb"; String username = "admin"; char[] password = "password".toCharArray(); startNewTransaction(); try { /** * Now go ahead and create our first transfer target */ transferService.createAndSaveTransferTarget(getMe, title, description, endpointProtocol, endpointHost, endpointPort, endpointPath, username, password); Set targets = transferService.getTransferTargets("Default Group"); assertTrue("targets is empty", targets.size() > 0); /** * Negative test - group does not exist */ targets = transferService.getTransferTargets("Rubbish"); assertTrue("targets is empty", targets.size() > 0); fail("group does not exist"); } catch (TransferException te) { // expect to go here } finally { endTransaction(); } } /** * */ public void testUpdateTransferTarget() throws Exception { startNewTransaction(); try { String updateMe = "updateMe"; String title = "title"; String description = "description"; String endpointProtocol = "http"; String endpointHost = "localhost"; int endpointPort = 8080; String endpointPath = "rhubarb"; String username = "admin"; char[] password = "password".toCharArray(); /** * Create our transfer target */ TransferTarget target = transferService.createAndSaveTransferTarget(updateMe, title, description, endpointProtocol, endpointHost, endpointPort, endpointPath, username, password); /* * Now update with exactly the same values. */ TransferTarget update1 = transferService.saveTransferTarget(target); assertNotNull("return value is null", update1); assertNotNull("node ref is null", update1.getNodeRef()); //titled aspect assertEquals("name not equal", update1.getName(), updateMe); assertEquals("title not equal", update1.getTitle(), title); assertEquals("description not equal", update1.getDescription(), description); // endpoint assertEquals("endpointProtocol not equal", update1.getEndpointProtocol(), endpointProtocol); assertEquals("endpointHost not equal", update1.getEndpointHost(), endpointHost); assertEquals("endpointPort not equal", update1.getEndpointPort(), endpointPort); assertEquals("endpointPath not equal", update1.getEndpointPath(), endpointPath); // authentication assertEquals("username not equal", update1.getUsername(), username); char[] pass = update1.getPassword(); assertEquals(password.length, pass.length); for(int i = 0; i < password.length; i++) { if(password[i] != pass[i]) { fail("password not equal:" + new String(password) + new String(pass)); } } /** * Now update with different values */ String title2 = "Two"; String description2 = "descriptionTwo"; String endpointProtocol2 = "https"; String endpointHost2 = "1.0.0.127"; int endpointPort2 = 4040; String endpointPath2 = "custard"; String username2 = "admin_two"; char[] password2 = "two".toCharArray(); target.setDescription(description2); target.setTitle(title2); target.setEndpointHost(endpointHost2); target.setEndpointPath(endpointPath2); target.setEndpointPort(endpointPort2); target.setEndpointProtocol(endpointProtocol2); target.setPassword(password2); target.setUsername(username2); TransferTarget update2 = transferService.saveTransferTarget(target); assertNotNull("return value is null", update2); assertNotNull("node ref is null", update2.getNodeRef()); //titled aspect assertEquals("name not equal", update2.getName(), updateMe); assertEquals("title not equal", update2.getTitle(), title2); assertEquals("description not equal", update2.getDescription(), description2); // endpoint assertEquals("endpointProtocol not equal", update2.getEndpointProtocol(), endpointProtocol2); assertEquals("endpointHost not equal", update2.getEndpointHost(), endpointHost2); assertEquals("endpointPort not equal", update2.getEndpointPort(), endpointPort2); assertEquals("endpointPath not equal", update2.getEndpointPath(), endpointPath2); // authentication assertEquals("username not equal", update2.getUsername(), username2); pass = update2.getPassword(); assertEquals(password2.length, pass.length); for(int i = 0; i < pass.length; i++) { if(pass[i] != password2[i]) { fail("password not equal:" + new String(pass) + new String(password2)); } } } finally { endTransaction(); } } /** * */ public void testDeleteTransferTarget() throws Exception { startNewTransaction(); try { String deleteMe = "deleteMe"; String title = "title"; String description = "description"; String endpointProtocol = "http"; String endpointHost = "localhost"; int endpointPort = 8080; String endpointPath = "rhubarb"; String username = "admin"; char[] password = "password".toCharArray(); /** * Now go ahead and create our first transfer target */ transferService.createAndSaveTransferTarget(deleteMe, title, description, endpointProtocol, endpointHost, endpointPort, endpointPath, username, password); transferService.deleteTransferTarget(deleteMe); /** * Negative test - try to delete the transfer target we deleted above. */ try { transferService.deleteTransferTarget(deleteMe); fail("duplicate name not detected"); } catch (TransferException e) { // expect to go here } /** * Negative test - try to delete a transfer target that has never existed */ try { transferService.deleteTransferTarget("rubbish"); fail("rubbish deleted"); } catch (TransferException e) { // expect to go here } } finally { endTransaction(); } } public void testEnableTransferTarget() throws Exception { startNewTransaction(); try { String targetName = "enableMe"; /** * Now go ahead and create our first transfer target */ TransferTarget enableMe = createTransferTarget(targetName); try { /** * Check a new target is enabled */ TransferTarget target = transferService.getTransferTarget(targetName); assertTrue("new target is not enabled", enableMe.isEnabled()); /** * Diasble the target */ transferService.enableTransferTarget(targetName, false); target = transferService.getTransferTarget(targetName); assertFalse("target is not disabled", target.isEnabled()); /** * Now re-enable the target */ transferService.enableTransferTarget(targetName, true); target = transferService.getTransferTarget(targetName); assertTrue("re-enabled target is not enabled", target.isEnabled()); } finally { transferService.deleteTransferTarget(targetName); } } finally { endTransaction(); } } /** * Test the transfer method by sending one node (CRUD). * * Step 1: Create a new node (No content) * transfer * * Step 2: Update Node title property * transfer * * Step 3: Update Content property * transfer * * Step 4: Transfer again * transfer (Should transfer but not request the content item) * * Step 5: Delete the node * * Step 6: Negative test : transfer no nodes * transfer (should throw exception) * * This is a unit test so it does some shenanigans to send to the same instance of alfresco. */ public void testTransferOneNode() throws Exception { setDefaultRollback(false); String CONTENT_TITLE = "ContentTitle"; String CONTENT_TITLE_UPDATED = "ContentTitleUpdated"; Locale CONTENT_LOCALE = Locale.GERMAN; String CONTENT_STRING = "Hello"; /** * For unit test * - replace the HTTP transport with the in-process transport * - replace the node factory with one that will map node refs, paths etc. * * Fake Repository Id */ TransferTransmitter transmitter = new UnitTestInProcessTransmitterImpl(receiver, contentService, transactionService); transferServiceImpl.setTransmitter(transmitter); UnitTestTransferManifestNodeFactory testNodeFactory = new UnitTestTransferManifestNodeFactory(this.transferManifestNodeFactory); transferServiceImpl.setTransferManifestNodeFactory(testNodeFactory); List> pathMap = testNodeFactory.getPathMap(); // Map company_home/guest_home to company_home so tranferred nodes and moved "up" one level. pathMap.add(new Pair(PathHelper.stringToPath(GUEST_HOME_XPATH_QUERY), PathHelper.stringToPath(COMPANY_HOME_XPATH_QUERY))); DescriptorService mockedDescriptorService = getMockDescriptorService(REPO_ID_A); transferServiceImpl.setDescriptorService(mockedDescriptorService); /** * Now go ahead and create our first transfer target */ String targetName = "testTransferOneNode"; TransferTarget transferMe; NodeRef contentNodeRef; NodeRef destNodeRef; startNewTransaction(); try { /** * Get guest home */ String guestHomeQuery = "/app:company_home/app:guest_home"; ResultSet guestHomeResult = searchService.query(StoreRef.STORE_REF_WORKSPACE_SPACESSTORE, SearchService.LANGUAGE_XPATH, guestHomeQuery); assertEquals("", 1, guestHomeResult.length()); NodeRef guestHome = guestHomeResult.getNodeRef(0); /** * Create a test node that we will read and write */ String name = GUID.generate(); ChildAssociationRef child = nodeService.createNode(guestHome, ContentModel.ASSOC_CONTAINS, QName.createQName(name), ContentModel.TYPE_CONTENT); contentNodeRef = child.getChildRef(); nodeService.setProperty(contentNodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(contentNodeRef, ContentModel.PROP_NAME, name); if(!transferService.targetExists(targetName)) { transferMe = createTransferTarget(targetName); } else { transferMe = transferService.getTransferTarget(targetName); } } finally { endTransaction(); } logger.debug("First transfer - create new node (no content yet)"); startNewTransaction(); try { /** * Step 1: Transfer our node which has no content */ { TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(contentNodeRef); definition.setNodes(nodes); transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source destNodeRef = testNodeFactory.getMappedNodeRef(contentNodeRef); assertFalse("unit test stuffed up - comparing with self", destNodeRef.equals(transferMe.getNodeRef())); assertTrue("dest node ref does not exist", nodeService.exists(destNodeRef)); assertEquals("title is wrong", (String)nodeService.getProperty(destNodeRef, ContentModel.PROP_TITLE), CONTENT_TITLE); assertEquals("type is wrong", nodeService.getType(contentNodeRef), nodeService.getType(destNodeRef)); // Check the modified time of the destination node is the same as the source node. Date destModifiedDate = (Date)nodeService.getProperty(destNodeRef, ContentModel.PROP_MODIFIED); Date srcModifiedDate = (Date)nodeService.getProperty(contentNodeRef, ContentModel.PROP_MODIFIED); logger.debug("srcModifiedDate : " + srcModifiedDate + " destModifiedDate : " + destModifiedDate); assertTrue("dest modified date is not correct", destModifiedDate.compareTo(srcModifiedDate)== 0); Date destCreatedDate = (Date)nodeService.getProperty(destNodeRef, ContentModel.PROP_CREATED); Date srcCreatedDate = (Date)nodeService.getProperty(contentNodeRef, ContentModel.PROP_CREATED); logger.debug("srcCreatedDate : " + srcCreatedDate + " destCreatedDate : " + destCreatedDate); assertTrue("dest created date is not correct", destCreatedDate.compareTo(srcCreatedDate)== 0); // Check injected transferred aspect. assertNotNull("transferredAspect", (String)nodeService.getProperty(destNodeRef, TransferModel.PROP_REPOSITORY_ID)); // Now set up the next test which is to change the title nodeService.setProperty(contentNodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE_UPDATED); } finally { endTransaction(); } logger.debug("Second transfer - update title property (no content yet)"); startNewTransaction(); try { /** * Step 2: * Transfer our node again - so this is an update of the title property */ { TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(contentNodeRef); definition.setNodes(nodes); transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source assertFalse("unit test stuffed up - comparing with self", destNodeRef.equals(transferMe.getNodeRef())); assertTrue("dest node ref does not exist", nodeService.exists(destNodeRef)); assertEquals("title is wrong", (String)nodeService.getProperty(destNodeRef, ContentModel.PROP_TITLE), CONTENT_TITLE_UPDATED); assertEquals("type is wrong", nodeService.getType(contentNodeRef), nodeService.getType(destNodeRef)); // Check the modified time of the destination node is the same as the source node. Date destModifiedDate = (Date)nodeService.getProperty(destNodeRef, ContentModel.PROP_MODIFIED); Date srcModifiedDate = (Date)nodeService.getProperty(contentNodeRef, ContentModel.PROP_MODIFIED); logger.debug("srcModifiedDate : " + srcModifiedDate + " destModifiedDate : " + destModifiedDate); assertTrue("after update, modified date is not correct", destModifiedDate.compareTo(srcModifiedDate) == 0); Date destCreatedDate = (Date)nodeService.getProperty(destNodeRef, ContentModel.PROP_CREATED); Date srcCreatedDate = (Date)nodeService.getProperty(contentNodeRef, ContentModel.PROP_CREATED); logger.debug("srcCreatedDate : " + srcCreatedDate + " destCreatedDate : " + destCreatedDate); assertTrue("after update, created date is not correct", destCreatedDate.compareTo(srcCreatedDate)== 0); // Check injected transferred aspect. assertNotNull("transferredAspect", (String)nodeService.getProperty(destNodeRef, TransferModel.PROP_REPOSITORY_ID)); ContentWriter writer = contentService.getWriter(contentNodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } finally { endTransaction(); } logger.debug("Transfer again - this is an update with new content"); startNewTransaction(); try { /** * Step 3: * Transfer our node again - so this is an update */ { TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(contentNodeRef); definition.setNodes(nodes); transferService.transfer(targetName, definition); } } finally { endTransaction(); } /** * Step 4: * Now transfer nothing - content items do not need to be transferred since its already on * the destination. */ logger.debug("Transfer again - with no new content"); startNewTransaction(); try { TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(contentNodeRef); definition.setNodes(nodes); transferService.transfer(targetName, definition); } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node still exists and in particular that the old content is still there assertFalse("unit test stuffed up - comparing with self", destNodeRef.equals(transferMe.getNodeRef())); assertTrue("dest node ref does not exist", nodeService.exists(destNodeRef)); assertEquals("title is wrong", (String)nodeService.getProperty(destNodeRef, ContentModel.PROP_TITLE), CONTENT_TITLE_UPDATED); assertEquals("type is wrong", nodeService.getType(contentNodeRef), nodeService.getType(destNodeRef)); // Check injected transferred aspect. assertNotNull("transferredAspect", (String)nodeService.getProperty(destNodeRef, TransferModel.PROP_REPOSITORY_ID)); } finally { endTransaction(); } /** * Step 5 * Delete the node through transfer of the archive node */ logger.debug("Transfer again - with no new content"); startNewTransaction(); try { nodeService.deleteNode(contentNodeRef); } finally { endTransaction(); } NodeRef deletedContentNodeRef = new NodeRef(StoreRef.STORE_REF_ARCHIVE_SPACESSTORE, contentNodeRef.getId()); startNewTransaction(); try { TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(deletedContentNodeRef); definition.setNodes(nodes); transferService.transfer(targetName, definition); } finally { endTransaction(); } startNewTransaction(); try { assertFalse("dest node still exists", nodeService.exists(destNodeRef)); } finally { endTransaction(); } /** * Step 6 * Negative test transfer nothing */ logger.debug("Transfer again - with no content - should throw exception"); try { TransferDefinition definition = new TransferDefinition(); transferService.transfer(targetName, definition); fail("exception not thrown"); } catch(TransferException te) { // expect to go here } } /** * Test the transfer method by sending a graph of nodes. * * This is a unit test so it does some shenanigans to send to he same instance of alfresco. */ public void testManyNodes() throws Exception { setDefaultRollback(false); String CONTENT_TITLE = "ContentTitle"; String CONTENT_TITLE_UPDATED = "ContentTitleUpdated"; String CONTENT_NAME = "Demo Node 1"; Locale CONTENT_LOCALE = Locale.GERMAN; String CONTENT_STRING = "The quick brown fox"; Setnodes = new HashSet(); String targetName = "testManyNodes"; NodeRef nodeA; NodeRef nodeB; NodeRef nodeAA; NodeRef nodeAB; NodeRef nodeABA; NodeRef nodeABB; NodeRef nodeABC; ChildAssociationRef child; /** * For unit test * - replace the HTTP transport with the in-process transport * - replace the node factory with one that will map node refs, paths etc. */ TransferTransmitter transmitter = new UnitTestInProcessTransmitterImpl(this.receiver, this.contentService, transactionService); transferServiceImpl.setTransmitter(transmitter); UnitTestTransferManifestNodeFactory testNodeFactory = new UnitTestTransferManifestNodeFactory(this.transferManifestNodeFactory); transferServiceImpl.setTransferManifestNodeFactory(testNodeFactory); List> pathMap = testNodeFactory.getPathMap(); // Map company_home/guest_home to company_home so tranferred nodes and moved "up" one level. pathMap.add(new Pair(PathHelper.stringToPath(GUEST_HOME_XPATH_QUERY), PathHelper.stringToPath(COMPANY_HOME_XPATH_QUERY))); DescriptorService mockedDescriptorService = getMockDescriptorService(REPO_ID_A); transferServiceImpl.setDescriptorService(mockedDescriptorService); TransferTarget transferMe; startNewTransaction(); try { /** * Get guest home */ String guestHomeQuery = "/app:company_home/app:guest_home"; ResultSet guestHomeResult = searchService.query(StoreRef.STORE_REF_WORKSPACE_SPACESSTORE, SearchService.LANGUAGE_XPATH, guestHomeQuery); assertEquals("", 1, guestHomeResult.length()); NodeRef guestHome = guestHomeResult.getNodeRef(0); /** * Create a test node that we will read and write */ String guid = GUID.generate(); /** * Create a tree * ManyNodesRoot * AC (Content Node) * A (Folder) * --AA * --AB (Folder) * ----ABA (Folder) * -------- 100+ nodes * ----ABB * ----ABC * B */ child = nodeService.createNode(guestHome, ContentModel.ASSOC_CONTAINS, QName.createQName(guid), ContentModel.TYPE_FOLDER); NodeRef testRootNode = child.getChildRef(); nodeService.setProperty(testRootNode , ContentModel.PROP_TITLE, guid); nodeService.setProperty(testRootNode , ContentModel.PROP_NAME, guid); nodes.add(testRootNode); child = nodeService.createNode(testRootNode, ContentModel.ASSOC_CONTAINS, QName.createQName("testNodeAC"), ContentModel.TYPE_CONTENT); NodeRef nodeAC = child.getChildRef(); nodeService.setProperty(nodeAC , ContentModel.PROP_TITLE, CONTENT_TITLE + "AC"); nodeService.setProperty(nodeAC , ContentModel.PROP_NAME, "DemoNodeAC"); { ContentWriter writer = contentService.getWriter(nodeAC , ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); nodes.add(nodeAC); } child = nodeService.createNode(testRootNode, ContentModel.ASSOC_CONTAINS, QName.createQName(NamespaceService.CONTENT_MODEL_1_0_URI,"testNodeA"), ContentModel.TYPE_FOLDER); nodeA = child.getChildRef(); nodeService.setProperty(nodeA , ContentModel.PROP_TITLE, "TestNodeA"); nodeService.setProperty(nodeA , ContentModel.PROP_NAME, "TestNodeA"); nodes.add(nodeA); child = nodeService.createNode(testRootNode, ContentModel.ASSOC_CONTAINS, QName.createQName(NamespaceService.CONTENT_MODEL_1_0_URI,"testNodeB"), ContentModel.TYPE_FOLDER); nodeB = child.getChildRef(); nodeService.setProperty(nodeB , ContentModel.PROP_TITLE, "TestNodeB"); nodeService.setProperty(nodeB , ContentModel.PROP_NAME, "TestNodeB"); nodes.add(nodeB); child = nodeService.createNode(nodeA, ContentModel.ASSOC_CONTAINS, QName.createQName(NamespaceService.CONTENT_MODEL_1_0_URI,"testNodeAA"), ContentModel.TYPE_FOLDER); nodeAA = child.getChildRef(); nodeService.setProperty(nodeAA , ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(nodeAA , ContentModel.PROP_NAME, "DemoNodeAA" ); nodes.add(nodeAA); child = nodeService.createNode(nodeA, ContentModel.ASSOC_CONTAINS, QName.createQName(NamespaceService.CONTENT_MODEL_1_0_URI,"testNodeAB"), ContentModel.TYPE_FOLDER); nodeAB = child.getChildRef(); nodeService.setProperty(nodeAB , ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(nodeAB , ContentModel.PROP_NAME, "DemoNodeAB" ); nodes.add(nodeAB); child = nodeService.createNode(nodeAB, ContentModel.ASSOC_CONTAINS, QName.createQName(NamespaceService.CONTENT_MODEL_1_0_URI,"testNodeABA"), ContentModel.TYPE_FOLDER); nodeABA = child.getChildRef(); nodeService.setProperty(nodeABA , ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(nodeABA , ContentModel.PROP_NAME, "DemoNodeABA" ); nodes.add(nodeABA); child = nodeService.createNode(nodeAB, ContentModel.ASSOC_CONTAINS, QName.createQName(NamespaceService.CONTENT_MODEL_1_0_URI,"testNodeABB"), ContentModel.TYPE_FOLDER); nodeABB = child.getChildRef(); nodeService.setProperty(nodeABB , ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(nodeABB , ContentModel.PROP_NAME, "DemoNodeABB" ); nodes.add(nodeABB); child = nodeService.createNode(nodeAB, ContentModel.ASSOC_CONTAINS, QName.createQName(NamespaceService.CONTENT_MODEL_1_0_URI,"testNodeABC"), ContentModel.TYPE_FOLDER); nodeABC = child.getChildRef(); nodeService.setProperty(nodeABC , ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(nodeABC , ContentModel.PROP_NAME, "DemoNodeABC" ); nodes.add(nodeABC); /** * Now go ahead and create our first transfer target */ if(!transferService.targetExists(targetName)) { transferMe = createTransferTarget(targetName); } else { transferMe = transferService.getTransferTarget(targetName); } } finally { endTransaction(); } startNewTransaction(); try { /** * Transfer our transfer target nodes */ { TransferDefinition definition = new TransferDefinition(); definition.setNodes(nodes); transferService.transfer(targetName, definition); } } finally { endTransaction(); } NodeRef destNodeA; NodeRef destNodeB; NodeRef destNodeAA; NodeRef destNodeAB; NodeRef destNodeABA; NodeRef destNodeABB; NodeRef destNodeABC; startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source destNodeA = testNodeFactory.getMappedNodeRef(nodeA); assertFalse("unit test stuffed up - comparing with self", destNodeA.equals(transferMe.getNodeRef())); assertTrue("dest node ref A does not exist", nodeService.exists(destNodeA)); assertEquals("title is wrong", (String)nodeService.getProperty(destNodeA, ContentModel.PROP_TITLE), "TestNodeA"); assertEquals("type is wrong", nodeService.getType(nodeA), nodeService.getType(destNodeA)); destNodeB = testNodeFactory.getMappedNodeRef(nodeB); assertTrue("dest node B does not exist", nodeService.exists(destNodeB)); destNodeAA = testNodeFactory.getMappedNodeRef(nodeAA); assertTrue("dest node AA ref does not exist", nodeService.exists(destNodeAA)); destNodeAB = testNodeFactory.getMappedNodeRef(nodeAB); assertTrue("dest node AB ref does not exist", nodeService.exists(destNodeAB)); destNodeABA = testNodeFactory.getMappedNodeRef(nodeABA); assertTrue("dest node ABA ref does not exist", nodeService.exists(destNodeABA)); destNodeABB = testNodeFactory.getMappedNodeRef(nodeABB); assertTrue("dest node ABB ref does not exist", nodeService.exists(destNodeABB)); destNodeABC = testNodeFactory.getMappedNodeRef(nodeABC); assertTrue("dest node ABC ref does not exist", nodeService.exists(destNodeABC)); } finally { endTransaction(); } startNewTransaction(); try { /** * Update a single node (NodeAB) from the middle of the tree */ { nodeService.setProperty(nodeAB , ContentModel.PROP_TITLE, CONTENT_TITLE_UPDATED); TransferDefinition definition = new TransferDefinition(); SettoUpdate = new HashSet(); toUpdate.add(nodeAB); definition.setNodes(toUpdate); transferService.transfer(targetName, definition); // assertEquals("title is wrong", (String)nodeService.getProperty(destNodeAB, ContentModel.PROP_TITLE), CONTENT_TITLE_UPDATED); } } finally { endTransaction(); } startNewTransaction(); try { /** * Now generate a large number of nodes */ for(int i = 0; i < 100; i++) { child = nodeService.createNode(nodeABA, ContentModel.ASSOC_CONTAINS, QName.createQName(GUID.generate() + i), ContentModel.TYPE_CONTENT); NodeRef nodeX = child.getChildRef(); nodeService.setProperty(nodeX , ContentModel.PROP_TITLE, CONTENT_TITLE + i); nodeService.setProperty(nodeX , ContentModel.PROP_NAME, CONTENT_NAME +i); nodes.add(nodeX); ContentWriter writer = contentService.getWriter(nodeX, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING + i); } } finally { endTransaction(); } startNewTransaction(); try { /** * Transfer our transfer target nodes */ { TransferDefinition definition = new TransferDefinition(); definition.setNodes(nodes); transferService.transfer(targetName, definition); } } finally { transferService.deleteTransferTarget(targetName); endTransaction(); } } // end many nodes /** * Test the path based update. * * This is a unit test so it does some shenanigans to send to the same instance of alfresco. */ public void testPathBasedUpdate() throws Exception { setDefaultRollback(false); /** * For unit test * - replace the HTTP transport with the in-process transport * - replace the node factory with one that will map node refs, paths etc. */ TransferTransmitter transmitter = new UnitTestInProcessTransmitterImpl(this.receiver, this.contentService, transactionService); transferServiceImpl.setTransmitter(transmitter); UnitTestTransferManifestNodeFactory testNodeFactory = new UnitTestTransferManifestNodeFactory(this.transferManifestNodeFactory); transferServiceImpl.setTransferManifestNodeFactory(testNodeFactory); List> pathMap = testNodeFactory.getPathMap(); // Map company_home/guest_home to company_home so tranferred nodes and moved "up" one level. pathMap.add(new Pair(PathHelper.stringToPath(GUEST_HOME_XPATH_QUERY), PathHelper.stringToPath(COMPANY_HOME_XPATH_QUERY))); DescriptorService mockedDescriptorService = getMockDescriptorService(REPO_ID_A); transferServiceImpl.setDescriptorService(mockedDescriptorService); String CONTENT_TITLE = "ContentTitle"; String CONTENT_TITLE_UPDATED = "ContentTitleUpdated"; String CONTENT_NAME = GUID.generate(); Locale CONTENT_LOCALE = Locale.GERMAN; String CONTENT_STRING = "Hello"; String targetName = GUID.generate(); NodeRef contentNodeRef; NodeRef newContentNodeRef; QName TEST_QNAME = QName.createQName(CONTENT_NAME); NodeRef guestHome; ChildAssociationRef child; startNewTransaction(); try { /** * Get guest home */ String guestHomeQuery = "/app:company_home/app:guest_home"; ResultSet guestHomeResult = searchService.query(StoreRef.STORE_REF_WORKSPACE_SPACESSTORE, SearchService.LANGUAGE_XPATH, guestHomeQuery); assertEquals("", 1, guestHomeResult.length()); guestHome = guestHomeResult.getNodeRef(0); /** * Create a test node that we will transfer. Its path is what is important */ child = nodeService.createNode(guestHome, ContentModel.ASSOC_CONTAINS, TEST_QNAME, ContentModel.TYPE_CONTENT); contentNodeRef = child.getChildRef(); nodeService.setProperty(contentNodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(contentNodeRef, ContentModel.PROP_NAME, CONTENT_NAME); ContentWriter writer = contentService.getWriter(contentNodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); /** * Now go ahead and create our first transfer target */ if(!transferService.targetExists(targetName)) { createTransferTarget(targetName); } else { transferService.getTransferTarget(targetName); } } finally { endTransaction(); } startNewTransaction(); try { /** * Transfer our transfer target node */ TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(contentNodeRef); definition.setNodes(nodes); transferService.transfer(targetName, definition); } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source NodeRef destNodeRef = testNodeFactory.getMappedNodeRef(contentNodeRef); assertFalse("unit test stuffed up - comparing with self", destNodeRef.equals(contentNodeRef)); assertTrue("dest node ref does not exist", nodeService.exists(destNodeRef)); assertEquals("title is wrong", (String)nodeService.getProperty(destNodeRef, ContentModel.PROP_TITLE), CONTENT_TITLE); assertEquals("type is wrong", nodeService.getType(contentNodeRef), nodeService.getType(destNodeRef)); /** * Now delete the content node and re-create another one with the old path */ nodeService.deleteNode(contentNodeRef); child = nodeService.createNode(guestHome, ContentModel.ASSOC_CONTAINS, TEST_QNAME, ContentModel.TYPE_CONTENT); newContentNodeRef = child.getChildRef(); nodeService.setProperty(newContentNodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE_UPDATED); /** * Transfer our node which is a new node (so will not exist on the back end) with a path that already has a node. */ { TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(newContentNodeRef); definition.setNodes(nodes); transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { NodeRef oldDestNodeRef = testNodeFactory.getMappedNodeRef(contentNodeRef); NodeRef newDestNodeRef = testNodeFactory.getMappedNodeRef(newContentNodeRef); // Now validate that the target node does not exist - it should have // been updated by path. assertFalse("unit test stuffed up - comparing with self", oldDestNodeRef.equals(newDestNodeRef)); assertFalse("new dest node ref exists", nodeService.exists(newDestNodeRef)); assertTrue("old dest node does not exists", nodeService.exists(oldDestNodeRef)); assertEquals("title is wrong", (String)nodeService.getProperty(oldDestNodeRef, ContentModel.PROP_TITLE), CONTENT_TITLE_UPDATED); // assertEquals("type is wrong", nodeService.getType(contentNodeRef), nodeService.getType(destNodeRef)); } finally { endTransaction(); } } // Path based update /** * Test the transfer method when it is running async. * * This is a unit test so it does some shenanigans to send to the same instance of alfresco. */ public void testAsyncCallback() throws Exception { int MAX_SLEEPS = 5; /** * Get guest home */ String guestHomeQuery = "/app:company_home/app:guest_home"; ResultSet guestHomeResult = searchService.query(StoreRef.STORE_REF_WORKSPACE_SPACESSTORE, SearchService.LANGUAGE_XPATH, guestHomeQuery); assertEquals("", 1, guestHomeResult.length()); NodeRef guestHome = guestHomeResult.getNodeRef(0); /** * For unit test * - replace the HTTP transport with the in-process transport * - replace the node factory with one that will map node refs, paths etc. */ TransferTransmitter transmitter = new UnitTestInProcessTransmitterImpl(this.receiver, this.contentService, transactionService); transferServiceImpl.setTransmitter(transmitter); UnitTestTransferManifestNodeFactory testNodeFactory = new UnitTestTransferManifestNodeFactory(this.transferManifestNodeFactory); transferServiceImpl.setTransferManifestNodeFactory(testNodeFactory); List> pathMap = testNodeFactory.getPathMap(); // Map company_home/guest_home to company_home so tranferred nodes and moved "up" one level. pathMap.add(new Pair(PathHelper.stringToPath(GUEST_HOME_XPATH_QUERY), PathHelper.stringToPath(COMPANY_HOME_XPATH_QUERY))); DescriptorService mockedDescriptorService = getMockDescriptorService(REPO_ID_A); transferServiceImpl.setDescriptorService(mockedDescriptorService); /** * Now go ahead and create our first transfer target * This needs to be committed before we can call transfer asycnc. */ String CONTENT_TITLE = "ContentTitle"; String CONTENT_NAME_A = "Demo Node A"; String CONTENT_NAME_B = "Demo Node B"; Locale CONTENT_LOCALE = Locale.GERMAN; String CONTENT_STRING = "Hello"; NodeRef nodeRefA = null; NodeRef nodeRefB = null; String targetName = "testAsyncCallback"; { UserTransaction trx = transactionService.getNonPropagatingUserTransaction(); trx.begin(); try { nodeRefA = nodeService.getChildByName(guestHome, ContentModel.ASSOC_CONTAINS, CONTENT_NAME_A); if(nodeRefA == null) { /** * Create a test node that we will read and write */ ChildAssociationRef child = nodeService.createNode(guestHome, ContentModel.ASSOC_CONTAINS, QName.createQName(GUID.generate()), ContentModel.TYPE_CONTENT); nodeRefA = child.getChildRef(); nodeService.setProperty(nodeRefA, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(nodeRefA, ContentModel.PROP_NAME, CONTENT_NAME_A); ContentWriter writer = contentService.getWriter(nodeRefA, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } nodeRefB = nodeService.getChildByName(guestHome, ContentModel.ASSOC_CONTAINS, CONTENT_NAME_B); if(nodeRefB == null) { ChildAssociationRef child = nodeService.createNode(guestHome, ContentModel.ASSOC_CONTAINS, QName.createQName(GUID.generate()), ContentModel.TYPE_CONTENT); nodeRefB = child.getChildRef(); nodeService.setProperty(nodeRefB, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(nodeRefB, ContentModel.PROP_NAME, CONTENT_NAME_B); ContentWriter writer = contentService.getWriter(nodeRefB, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } /** * Now go ahead and create our first transfer target */ if(!transferService.targetExists(targetName)) { createTransferTarget(targetName); } else { transferService.getTransferTarget(targetName); } } finally { trx.commit(); } } /** * The transfer report is a plain report of the transfer - no async shenanigans to worry about */ ListtransferReport = new ArrayList(50); startNewTransaction(); try { /** * Call the transferAsync method. */ { TestTransferCallback callback = new TestTransferCallback(); Set callbacks = new HashSet(); callbacks.add(callback); TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(nodeRefA); nodes.add(nodeRefB); definition.setNodes(nodes); transferService.transferAsync(targetName, definition, callbacks); logger.debug("transfer async has returned"); /** * Need to poll the transfer events here until callback receives the last event */ Queue events = callback.getEvents(); int sleepCount = MAX_SLEEPS; boolean ended = false; TransferEvent event = events.poll(); while (!ended) { logger.debug("polling loop:" + sleepCount); while(event != null) { /** * Got an event - reset the sleep counter */ sleepCount = MAX_SLEEPS; logger.debug("Got an event" + event.toString()); /** * Squirrel away the event for analysis later */ transferReport.add(event); /** * If we read the last record which will either be SUCCESS or ERROR then we we have finished */ if(event.isLast()) { logger.debug("got last event"); ended = true; } /** * Try to get the next event */ event = events.poll(); } if(event == null && !ended) { if(sleepCount <= 0) { fail("timed out without receiving last event"); ended = true; } else { /** * No content - go to sleep to wait for some more */ if(sleepCount-- > 0) { // Sleep for 5 second Thread.sleep(5000); } } /** * Try to get the next event */ event = events.poll(); } } } /** * Now validate the transferReport */ assertTrue("transfer report is too small", transferReport.size() > 2); assertTrue("transfer report does not start with START", transferReport.get(0).getTransferState().equals(TransferEvent.TransferState.START)); assertTrue("transfer report does not end with SUCCESS", transferReport.get(transferReport.size()-1).getTransferState().equals(TransferEvent.TransferState.SUCCESS)); } finally { // UserTransaction trx = transactionService.getNonPropagatingUserTransaction(); // trx.begin(); transferService.deleteTransferTarget(targetName); // trx.commit(); endTransaction(); } } // test async callback /** * Test the transfer cancel method when it is running async. * * This is a unit test so it does some shenanigans to send to the same instance of alfresco. */ public void testAsyncCancel() throws Exception { int MAX_SLEEPS = 5; /** * Get guest home */ String guestHomeQuery = "/app:company_home/app:guest_home"; ResultSet guestHomeResult = searchService.query(StoreRef.STORE_REF_WORKSPACE_SPACESSTORE, SearchService.LANGUAGE_XPATH, guestHomeQuery); assertEquals("", 1, guestHomeResult.length()); NodeRef guestHome = guestHomeResult.getNodeRef(0); /** * For unit test * - replace the HTTP transport with the in-process transport * - replace the node factory with one that will map node refs, paths etc. */ TransferTransmitter transmitter = new UnitTestInProcessTransmitterImpl(this.receiver, this.contentService, transactionService); transferServiceImpl.setTransmitter(transmitter); UnitTestTransferManifestNodeFactory testNodeFactory = new UnitTestTransferManifestNodeFactory(this.transferManifestNodeFactory); transferServiceImpl.setTransferManifestNodeFactory(testNodeFactory); List> pathMap = testNodeFactory.getPathMap(); // Map company_home/guest_home to company_home so tranferred nodes and moved "up" one level. pathMap.add(new Pair(PathHelper.stringToPath(GUEST_HOME_XPATH_QUERY), PathHelper.stringToPath(COMPANY_HOME_XPATH_QUERY))); DescriptorService mockedDescriptorService = getMockDescriptorService(REPO_ID_A); transferServiceImpl.setDescriptorService(mockedDescriptorService); /** * Now go ahead and create our first transfer target * This needs to be committed before we can call transfer asycnc. */ String CONTENT_TITLE = "ContentTitle"; String CONTENT_NAME_A = "Demo Node A"; String CONTENT_NAME_B = "Demo Node B"; Locale CONTENT_LOCALE = Locale.GERMAN; String CONTENT_STRING = "Hello"; NodeRef nodeRefA = null; NodeRef nodeRefB = null; String targetName = "testAsyncCallback"; { UserTransaction trx = transactionService.getNonPropagatingUserTransaction(); trx.begin(); try { nodeRefA = nodeService.getChildByName(guestHome, ContentModel.ASSOC_CONTAINS, CONTENT_NAME_A); if(nodeRefA == null) { /** * Create a test node that we will read and write */ ChildAssociationRef child = nodeService.createNode(guestHome, ContentModel.ASSOC_CONTAINS, QName.createQName(GUID.generate()), ContentModel.TYPE_CONTENT); nodeRefA = child.getChildRef(); nodeService.setProperty(nodeRefA, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(nodeRefA, ContentModel.PROP_NAME, CONTENT_NAME_A); ContentWriter writer = contentService.getWriter(nodeRefA, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } nodeRefB = nodeService.getChildByName(guestHome, ContentModel.ASSOC_CONTAINS, CONTENT_NAME_B); if(nodeRefB == null) { ChildAssociationRef child = nodeService.createNode(guestHome, ContentModel.ASSOC_CONTAINS, QName.createQName(GUID.generate()), ContentModel.TYPE_CONTENT); nodeRefB = child.getChildRef(); nodeService.setProperty(nodeRefB, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(nodeRefB, ContentModel.PROP_NAME, CONTENT_NAME_B); ContentWriter writer = contentService.getWriter(nodeRefB, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } /** * Now go ahead and create our first transfer target */ if(!transferService.targetExists(targetName)) { createTransferTarget(targetName); } else { transferService.getTransferTarget(targetName); } } finally { trx.commit(); } } /** * The transfer report is a plain report of the transfer - no async shenanigans to worry about */ ListtransferReport = new ArrayList(50); startNewTransaction(); try { /** * Call the transferAsync method. */ { /** * The poison callback will cancel the transfer after * the begin */ TransferCallback poison = new TransferCallback() { String transferId = null; public void processEvent(TransferEvent event) { logger.debug(event.toString()); if(event instanceof TransferEventBegin) { TransferEventBegin beginEvent = (TransferEventBegin)event; transferId = beginEvent.getTransferId(); transferService.cancelAsync(transferId); } } }; TestTransferCallback callback = new TestTransferCallback(); Set callbacks = new HashSet(); callbacks.add(callback); callbacks.add(poison); TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(nodeRefA); nodes.add(nodeRefB); definition.setNodes(nodes); transferService.transferAsync(targetName, definition, callbacks); logger.debug("transfer async has returned"); /** * Need to poll the transfer events here until callback receives the last event */ Queue events = callback.getEvents(); int sleepCount = MAX_SLEEPS; boolean ended = false; TransferEvent event = events.poll(); while (!ended) { logger.debug("polling loop:" + sleepCount); while(event != null) { /** * Got an event - reset the sleep counter */ sleepCount = MAX_SLEEPS; logger.debug("Got an event" + event.toString()); /** * Squirrel away the event for analysis later */ transferReport.add(event); /** * If we read the last record which will either be SUCCESS or ERROR then we we have finished */ if(event.isLast()) { logger.debug("got last event"); ended = true; } /** * Try to get the next event */ event = events.poll(); } if(event == null && !ended) { if(sleepCount <= 0) { fail("timed out without receiving last event"); ended = true; } else { /** * No content - go to sleep to wait for some more */ if(sleepCount-- > 0) { // Sleep for 5 second Thread.sleep(5000); } } /** * Try to get the next event */ event = events.poll(); } } } /** * Now validate the transferReport */ assertTrue("transfer report is too small", transferReport.size() > 2); assertTrue("transfer report does not start with START", transferReport.get(0).getTransferState().equals(TransferEvent.TransferState.START)); assertTrue("transfer report does not end with ERROR", transferReport.get(transferReport.size()-1).getTransferState().equals(TransferEvent.TransferState.ERROR)); } finally { // UserTransaction trx = transactionService.getNonPropagatingUserTransaction(); // trx.begin(); transferService.deleteTransferTarget(targetName); // trx.commit(); endTransaction(); } } // test async cancel /** * Test the transfer report. * * This is a unit test so it does some shenanigans to send to the same instance of alfresco. */ public void testTransferReport() throws Exception { setDefaultRollback(false); /** * Get guest home */ String guestHomeQuery = "/app:company_home/app:guest_home"; ResultSet guestHomeResult = searchService.query(StoreRef.STORE_REF_WORKSPACE_SPACESSTORE, SearchService.LANGUAGE_XPATH, guestHomeQuery); assertEquals("", 1, guestHomeResult.length()); NodeRef guestHome = guestHomeResult.getNodeRef(0); /** * For unit test * - replace the HTTP transport with the in-process transport * - replace the node factory with one that will map node refs, paths etc. */ TransferTransmitter transmitter = new UnitTestInProcessTransmitterImpl(this.receiver, this.contentService, transactionService); transferServiceImpl.setTransmitter(transmitter); UnitTestTransferManifestNodeFactory testNodeFactory = new UnitTestTransferManifestNodeFactory(this.transferManifestNodeFactory); transferServiceImpl.setTransferManifestNodeFactory(testNodeFactory); List> pathMap = testNodeFactory.getPathMap(); // Map company_home/guest_home to company_home so tranferred nodes and moved "up" one level. pathMap.add(new Pair(PathHelper.stringToPath(GUEST_HOME_XPATH_QUERY), PathHelper.stringToPath(COMPANY_HOME_XPATH_QUERY))); DescriptorService mockedDescriptorService = getMockDescriptorService(REPO_ID_A); transferServiceImpl.setDescriptorService(mockedDescriptorService); /** * Now go ahead and create our first transfer target * This needs to be committed before we can call transfer asycnc. */ String CONTENT_TITLE = "ContentTitle"; String CONTENT_NAME_A = "Demo Node A"; String CONTENT_NAME_B = "Demo Node B"; Locale CONTENT_LOCALE = Locale.GERMAN; String CONTENT_STRING = "Hello"; NodeRef nodeRefA = null; NodeRef nodeRefB = null; String targetName = "testTransferReport"; startNewTransaction(); try { nodeRefA = nodeService.getChildByName(guestHome, ContentModel.ASSOC_CONTAINS, CONTENT_NAME_A); if(nodeRefA == null) { /** * Create a test node that we will read and write */ ChildAssociationRef child = nodeService.createNode(guestHome, ContentModel.ASSOC_CONTAINS, QName.createQName(GUID.generate()), ContentModel.TYPE_CONTENT); nodeRefA = child.getChildRef(); nodeService.setProperty(nodeRefA, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(nodeRefA, ContentModel.PROP_NAME, CONTENT_NAME_A); ContentWriter writer = contentService.getWriter(nodeRefA, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } nodeRefB = nodeService.getChildByName(guestHome, ContentModel.ASSOC_CONTAINS, CONTENT_NAME_B); if(nodeRefB == null) { ChildAssociationRef child = nodeService.createNode(guestHome, ContentModel.ASSOC_CONTAINS, QName.createQName(GUID.generate()), ContentModel.TYPE_CONTENT); nodeRefB = child.getChildRef(); nodeService.setProperty(nodeRefB, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(nodeRefB, ContentModel.PROP_NAME, CONTENT_NAME_B); ContentWriter writer = contentService.getWriter(nodeRefB, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } /** * Now go ahead and create our first transfer target */ if(!transferService.targetExists(targetName)) { createTransferTarget(targetName); } } finally { endTransaction(); } NodeRef transferReport = null; startNewTransaction(); try { /** * Call the transfer method. */ { TestTransferCallback callback = new TestTransferCallback(); Set callbacks = new HashSet(); callbacks.add(callback); TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(nodeRefA); nodes.add(nodeRefB); definition.setNodes(nodes); transferReport = transferService.transfer(targetName, definition, callbacks); assertNotNull("transfer report is null", transferReport); // Can't dirty read transfer report here } } finally { endTransaction(); } startNewTransaction(); try { ContentReader reader = contentService.getReader(transferReport, ContentModel.PROP_CONTENT); assertNotNull("transfer reader is null", reader); ContentReader reader2 = contentService.getReader(transferReport, ContentModel.PROP_CONTENT); assertNotNull("transfer reader is null", reader); logger.debug("now show the contents of the transfer report"); reader2.getContent(System.out); // Now validate the client side transfer report against the XSD Source transferReportSource = new StreamSource(reader.getContentInputStream()); SchemaFactory sf = SchemaFactory.newInstance(XMLConstants.W3C_XML_SCHEMA_NS_URI); final String TRANSFER_REPORT_SCHEMA_LOCATION = "classpath:org/alfresco/repo/transfer/report/TransferReport.xsd"; Schema schema = sf.newSchema(ResourceUtils.getURL(TRANSFER_REPORT_SCHEMA_LOCATION)); Validator validator = schema.newValidator(); try { validator.validate(transferReportSource); } catch (Exception e) { fail(e.getMessage() ); } logger.debug("now delete the target:" + targetName); transferService.deleteTransferTarget(targetName); } finally { endTransaction(); } } // test transfer report // /** // * Test the transfer method with big content - commented out since it takes a long time to run. // */ // public void testTransferOneNodeWithBigContent() throws Exception // { // String CONTENT_TITLE = "ContentTitle"; // String CONTENT_NAME = "Demo Node 6"; // Locale CONTENT_LOCALE = Locale.GERMAN; // String CONTENT_STRING = "Hello"; // // String targetName = "testTransferOneNodeWithBigContent"; // // String guestHomeQuery = "/app:company_home/app:guest_home"; // ResultSet result = searchService.query(StoreRef.STORE_REF_WORKSPACE_SPACESSTORE, SearchService.LANGUAGE_XPATH, guestHomeQuery); // // assertEquals("", 1, result.length()); // NodeRef guestHome = result.getNodeRef(0); // ChildAssociationRef childAssoc = result.getChildAssocRef(0); // System.out.println("Guest home:" + guestHome); // assertNotNull(guestHome); // // /** // * Now go ahead and create our first transfer target // */ // TransferTarget transferMe = createTransferTarget(targetName); // // /** // * Create a test node that we will read and write // */ // ChildAssociationRef child = nodeService.createNode(guestHome, ContentModel.ASSOC_CONTAINS, QName.createQName("testNode6"), ContentModel.TYPE_CONTENT); // // // File tempFile = TempFileProvider.createTempFile("test", ".dat"); // FileWriter fw = new FileWriter(tempFile); // for(int i = 0; i < 100000000; i++) // { // fw.write("hello world this is my text, I wonder how much text I can transfer?" + i); // } // System.out.println("Temp File Size is:" + tempFile.length()); // fw.close(); // // NodeRef contentNodeRef = child.getChildRef(); // ContentWriter writer = contentService.getWriter(contentNodeRef, ContentModel.PROP_CONTENT, true); // writer.setLocale(CONTENT_LOCALE); // //File file = new File("c:/temp/images/BigCheese1.bmp"); // writer.setMimetype("application/data"); // //writer.putContent(file); // writer.putContent(tempFile); // // tempFile.delete(); // // nodeService.setProperty(contentNodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); // nodeService.setProperty(contentNodeRef, ContentModel.PROP_NAME, CONTENT_NAME); // // try // { // /** // * Transfer the node created above // */ // { // TransferDefinition definition = new TransferDefinition(); // Setnodes = new HashSet(); // nodes.add(contentNodeRef); // definition.setNodes(nodes); // transferService.transfer(targetName, definition, null); // } // // /** // * Negative test transfer nothing // */ // try // { // TransferDefinition definition = new TransferDefinition(); // transferService.transfer(targetName, definition, null); // fail("exception not thrown"); // } // catch(TransferException te) // { // // expect to go here // } // } // finally // { // transferService.deleteTransferTarget(targetName); // } // } /** * Test the transfer method behaviour with respect to sync folders - sending a complete set * of nodes and implying delete from the absence of an association. * * This is a unit test so it does some shenanigans to send to the same instance of alfresco. * * Tree of nodes * * A1 * | | * A2 A3 (Content Node) * | * A4 A5 (Content Node) * * Test steps - * 1 add A1 * transfer(sync) * 2 add A2, A3, A4, A5 * transfer(sync) * 3 remove A2 * transfer(sync) A4 and A5 should cascade delete on source * 4 remove A3 * transfer(sync) * 5 add back A3 - new node ref * transfer(sync) * 6 add A2, A4, A5 * transfer(sync) * 7 test delete and restore of a single node * remove A3 . * transfer * restore node A3 * transfer * 8 test delete and restore of a tree of nodes * remove A2 (A4 and A5 should cascade delete on source as well) * transfer * restore node A2 (and A4 and A5 cascade restore) * transfer */ public void testTransferSyncNodes() throws Exception { setDefaultRollback(false); String CONTENT_TITLE = "ContentTitle"; String CONTENT_TITLE_UPDATED = "ContentTitleUpdated"; Locale CONTENT_LOCALE = Locale.GERMAN; String CONTENT_STRING = "Hello"; /** * For unit test * - replace the HTTP transport with the in-process transport * - replace the node factory with one that will map node refs, paths etc. */ TransferTransmitter transmitter = new UnitTestInProcessTransmitterImpl(receiver, contentService, transactionService); transferServiceImpl.setTransmitter(transmitter); UnitTestTransferManifestNodeFactory testNodeFactory = new UnitTestTransferManifestNodeFactory(this.transferManifestNodeFactory); transferServiceImpl.setTransferManifestNodeFactory(testNodeFactory); List> pathMap = testNodeFactory.getPathMap(); // Map company_home/guest_home to company_home so tranferred nodes and moved "up" one level. pathMap.add(new Pair(PathHelper.stringToPath(GUEST_HOME_XPATH_QUERY), PathHelper.stringToPath(COMPANY_HOME_XPATH_QUERY))); DescriptorService mockedDescriptorService = getMockDescriptorService(REPO_ID_A); transferServiceImpl.setDescriptorService(mockedDescriptorService); /** * Now go ahead and create our first transfer target */ String targetName = "testTransferSyncNodes"; TransferTarget transferMe; NodeRef A1NodeRef; NodeRef A2NodeRef; NodeRef A3NodeRef; NodeRef A4NodeRef; NodeRef A5NodeRef; NodeRef destNodeRef; startNewTransaction(); try { /** * Get guest home */ String guestHomeQuery = "/app:company_home/app:guest_home"; ResultSet guestHomeResult = searchService.query(StoreRef.STORE_REF_WORKSPACE_SPACESSTORE, SearchService.LANGUAGE_XPATH, guestHomeQuery); assertEquals("", 1, guestHomeResult.length()); NodeRef guestHome = guestHomeResult.getNodeRef(0); /** * Create a test nodes A1 through A5 that we will read and write */ { // Node A1 String name = GUID.generate(); ChildAssociationRef child = nodeService.createNode(guestHome, ContentModel.ASSOC_CONTAINS, QName.createQName(name), ContentModel.TYPE_FOLDER); A1NodeRef = child.getChildRef(); nodeService.setProperty(A1NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(A1NodeRef, ContentModel.PROP_NAME, name); } { // Node A2 ChildAssociationRef child = nodeService.createNode(A1NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A2"), ContentModel.TYPE_FOLDER); A2NodeRef = child.getChildRef(); nodeService.setProperty(A2NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(A2NodeRef, ContentModel.PROP_NAME, "A2"); } { // Node A3 ChildAssociationRef child = nodeService.createNode(A1NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A3"), ContentModel.TYPE_CONTENT); A3NodeRef = child.getChildRef(); nodeService.setProperty(A3NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(A3NodeRef, ContentModel.PROP_NAME, "A3"); ContentWriter writer = contentService.getWriter(A3NodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } { // Node A4 ChildAssociationRef child = nodeService.createNode(A2NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A4"), ContentModel.TYPE_CONTENT); A4NodeRef = child.getChildRef(); nodeService.setProperty(A4NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(A4NodeRef, ContentModel.PROP_NAME, "A4"); ContentWriter writer = contentService.getWriter(A4NodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } { // Node A5 ChildAssociationRef child = nodeService.createNode(A2NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A5"), ContentModel.TYPE_CONTENT); A5NodeRef = child.getChildRef(); nodeService.setProperty(A5NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(A5NodeRef, ContentModel.PROP_NAME, "A5"); ContentWriter writer = contentService.getWriter(A5NodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } // Create the transfer target if it does not already exist if(!transferService.targetExists(targetName)) { transferMe = createTransferTarget(targetName); } else { transferMe = transferService.getTransferTarget(targetName); } } finally { endTransaction(); } /** * Step 1. Add Node A1. */ startNewTransaction(); try { /** * Transfer Node A with no children */ { TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(A1NodeRef); definition.setNodes(nodes); definition.setSync(true); transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source destNodeRef = testNodeFactory.getMappedNodeRef(A1NodeRef); assertFalse("unit test stuffed up - comparing with self", destNodeRef.equals(transferMe.getNodeRef())); assertTrue("dest node ref does not exist", nodeService.exists(destNodeRef)); assertEquals("title is wrong", (String)nodeService.getProperty(destNodeRef, ContentModel.PROP_TITLE), CONTENT_TITLE); assertEquals("type is wrong", nodeService.getType(A1NodeRef), nodeService.getType(destNodeRef)); // Check injected transferred aspect. assertNotNull("transferredAspect", (String)nodeService.getProperty(destNodeRef, TransferModel.PROP_REPOSITORY_ID)); } finally { endTransaction(); } /** * Step 2. Add Node A2, A3, A4, A5. */ startNewTransaction(); try { /** * Transfer Node A 1-5 */ { TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(A1NodeRef); nodes.add(A2NodeRef); nodes.add(A3NodeRef); nodes.add(A4NodeRef); nodes.add(A5NodeRef); definition.setNodes(nodes); definition.setSync(true); transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source destNodeRef = testNodeFactory.getMappedNodeRef(A1NodeRef); assertFalse("unit test stuffed up - comparing with self", destNodeRef.equals(transferMe.getNodeRef())); assertTrue("dest node ref A1 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A1NodeRef))); assertTrue("dest node ref A2 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A2NodeRef))); assertTrue("dest node ref A3 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A3NodeRef))); assertTrue("dest node ref A4 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A4NodeRef))); assertTrue("dest node ref A5 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A5NodeRef))); // Check injected transferred aspect. assertNotNull("transferredAspect", (String)nodeService.getProperty(destNodeRef, TransferModel.PROP_REPOSITORY_ID)); } finally { endTransaction(); } /** * Step 3 - remove folder node A2 */ startNewTransaction(); try { nodeService.deleteNode(A2NodeRef); } finally { endTransaction(); } startNewTransaction(); try { /** * Transfer Node A 1-5 */ { TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(A1NodeRef); //nodes.add(A2NodeRef); nodes.add(A3NodeRef); //nodes.add(A4NodeRef); //nodes.add(A5NodeRef); definition.setNodes(nodes); definition.setSync(true); transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source destNodeRef = testNodeFactory.getMappedNodeRef(A1NodeRef); assertFalse("unit test stuffed up - comparing with self", destNodeRef.equals(transferMe.getNodeRef())); assertTrue("dest node ref A1 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A1NodeRef))); assertFalse("dest node ref A2 has not been deleted", nodeService.exists(testNodeFactory.getMappedNodeRef(A2NodeRef))); assertTrue("dest node ref A3 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A3NodeRef))); assertFalse("dest node ref A4 has not been deleted", nodeService.exists(testNodeFactory.getMappedNodeRef(A4NodeRef))); assertFalse("dest node ref A5 has not been deleted", nodeService.exists(testNodeFactory.getMappedNodeRef(A5NodeRef))); // Check injected transferred aspect. assertNotNull("transferredAspect", (String)nodeService.getProperty(destNodeRef, TransferModel.PROP_REPOSITORY_ID)); } finally { endTransaction(); } /** * Step 4 - remove content node A3 */ startNewTransaction(); try { nodeService.deleteNode(A3NodeRef); } finally { endTransaction(); } startNewTransaction(); try { /** * Transfer Node A 1-5 */ { TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(A1NodeRef); //nodes.add(A2NodeRef); //nodes.add(A3NodeRef); //nodes.add(A4NodeRef); //nodes.add(A5NodeRef); definition.setNodes(nodes); definition.setSync(true); transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source destNodeRef = testNodeFactory.getMappedNodeRef(A1NodeRef); assertFalse("unit test stuffed up - comparing with self", destNodeRef.equals(transferMe.getNodeRef())); assertTrue("dest node ref A1 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A1NodeRef))); assertFalse("dest node ref A2 has not been deleted", nodeService.exists(testNodeFactory.getMappedNodeRef(A2NodeRef))); assertFalse("dest node ref A3 has not been deleted", nodeService.exists(testNodeFactory.getMappedNodeRef(A3NodeRef))); assertFalse("dest node ref A4 has not been deleted", nodeService.exists(testNodeFactory.getMappedNodeRef(A4NodeRef))); assertFalse("dest node ref A5 has not been deleted", nodeService.exists(testNodeFactory.getMappedNodeRef(A5NodeRef))); // Check injected transferred aspect. assertNotNull("transferredAspect", (String)nodeService.getProperty(destNodeRef, TransferModel.PROP_REPOSITORY_ID)); } finally { endTransaction(); } /** * Step 5. Add back A3. */ startNewTransaction(); try { ChildAssociationRef child = nodeService.createNode(A1NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A3"), ContentModel.TYPE_CONTENT); A3NodeRef = child.getChildRef(); nodeService.setProperty(A3NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(A3NodeRef, ContentModel.PROP_NAME, "A3"); ContentWriter writer = contentService.getWriter(A3NodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } finally { endTransaction(); } startNewTransaction(); try { /** * Transfer Node A 1-5 */ { TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(A1NodeRef); //nodes.add(A2NodeRef); nodes.add(A3NodeRef); //nodes.add(A4NodeRef); //nodes.add(A5NodeRef); definition.setNodes(nodes); definition.setSync(true); transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source destNodeRef = testNodeFactory.getMappedNodeRef(A1NodeRef); assertFalse("unit test stuffed up - comparing with self", destNodeRef.equals(transferMe.getNodeRef())); assertTrue("dest node ref A1 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A1NodeRef))); assertFalse("dest node ref A2 has not been deleted", nodeService.exists(testNodeFactory.getMappedNodeRef(A2NodeRef))); assertTrue("dest node A3 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A3NodeRef))); assertFalse("dest node ref A4 has not been deleted", nodeService.exists(testNodeFactory.getMappedNodeRef(A4NodeRef))); assertFalse("dest node ref A5 has not been deleted", nodeService.exists(testNodeFactory.getMappedNodeRef(A5NodeRef))); // Check injected transferred aspect. assertNotNull("transferredAspect", (String)nodeService.getProperty(destNodeRef, TransferModel.PROP_REPOSITORY_ID)); } finally { endTransaction(); } /** * Step 6. add A2, A4, A5 */ startNewTransaction(); try { { // Node A2 ChildAssociationRef child = nodeService.createNode(A1NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A2"), ContentModel.TYPE_FOLDER); A2NodeRef = child.getChildRef(); nodeService.setProperty(A2NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(A2NodeRef, ContentModel.PROP_NAME, "A2"); } { // Node A4 ChildAssociationRef child = nodeService.createNode(A2NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A4"), ContentModel.TYPE_CONTENT); A4NodeRef = child.getChildRef(); nodeService.setProperty(A4NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(A4NodeRef, ContentModel.PROP_NAME, "A4"); ContentWriter writer = contentService.getWriter(A4NodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } { // Node A5 ChildAssociationRef child = nodeService.createNode(A2NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A5"), ContentModel.TYPE_CONTENT); A5NodeRef = child.getChildRef(); nodeService.setProperty(A5NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(A5NodeRef, ContentModel.PROP_NAME, "A5"); ContentWriter writer = contentService.getWriter(A5NodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } } finally { endTransaction(); } startNewTransaction(); try { /** * Transfer Node A 1-5 */ { TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(A1NodeRef); nodes.add(A2NodeRef); nodes.add(A3NodeRef); nodes.add(A4NodeRef); nodes.add(A5NodeRef); definition.setNodes(nodes); definition.setSync(true); transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source destNodeRef = testNodeFactory.getMappedNodeRef(A1NodeRef); assertFalse("unit test stuffed up - comparing with self", destNodeRef.equals(transferMe.getNodeRef())); assertTrue("dest node A1 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A1NodeRef))); assertTrue("dest node A2 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A2NodeRef))); assertTrue("dest node A3 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A3NodeRef))); assertTrue("dest node A4 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A4NodeRef))); assertTrue("dest node A5 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A5NodeRef))); // Check injected transferred aspect. assertNotNull("transferredAspect", (String)nodeService.getProperty(destNodeRef, TransferModel.PROP_REPOSITORY_ID)); } finally { endTransaction(); } /** * Step 7 - test delete and restore of a single node * remove A3 . * transfer * restore node A3 * transfer */ startNewTransaction(); try { logger.debug("Step 7 - delete node A3"); nodeService.deleteNode(A3NodeRef); } finally { endTransaction(); } startNewTransaction(); try { /** * Transfer Node A 1-5 */ { TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(A1NodeRef); nodes.add(A2NodeRef); //nodes.add(A3NodeRef); A3 has gone. nodes.add(A4NodeRef); nodes.add(A5NodeRef); definition.setNodes(nodes); definition.setSync(true); transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { assertTrue("dest node ref A1 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A1NodeRef))); assertTrue("dest node ref A2 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A2NodeRef))); assertFalse("dest node ref A3 has not been deleted", nodeService.exists(testNodeFactory.getMappedNodeRef(A3NodeRef))); assertTrue("dest node ref A4 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A4NodeRef))); assertTrue("dest node ref A5 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A5NodeRef))); } finally { endTransaction(); } startNewTransaction(); try { NodeRef archivedNode = new NodeRef(StoreRef.STORE_REF_ARCHIVE_SPACESSTORE, A3NodeRef.getId()); NodeRef newNodeRef = nodeService.restoreNode(archivedNode, A1NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A3")); assertEquals("restored node ref is different", newNodeRef, A3NodeRef); logger.debug("Step 7 - restore node A3"); } finally { endTransaction(); } startNewTransaction(); try { /** * Transfer Node A 1-5. * (So we are seeing what happens to node 3 on the target */ { TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(A1NodeRef); nodes.add(A2NodeRef); nodes.add(A3NodeRef); nodes.add(A4NodeRef); nodes.add(A5NodeRef); definition.setNodes(nodes); definition.setSync(true); transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source assertFalse("unit test stuffed up - comparing with self", destNodeRef.equals(transferMe.getNodeRef())); assertTrue("dest node ref A1 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A1NodeRef))); assertTrue("dest node ref A2 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A2NodeRef))); assertTrue("dest node ref A3 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A3NodeRef))); assertTrue("dest node ref A4 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A4NodeRef))); assertTrue("dest node ref A5 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A5NodeRef))); } finally { endTransaction(); } /** * Step 8 - test delete and restore of a tree * remove A2 (A4, A5) should cascade delete. * transfer * restore node A2 * transfer */ startNewTransaction(); try { nodeService.deleteNode(A2NodeRef); } finally { endTransaction(); } startNewTransaction(); try { /** * Transfer Node A 1-5 */ { TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(A1NodeRef); //nodes.add(A2NodeRef); nodes.add(A3NodeRef); //nodes.add(A4NodeRef); //nodes.add(A5NodeRef); definition.setNodes(nodes); definition.setSync(true); transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source destNodeRef = testNodeFactory.getMappedNodeRef(A1NodeRef); assertFalse("unit test stuffed up - comparing with self", destNodeRef.equals(transferMe.getNodeRef())); assertTrue("dest node ref A1 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A1NodeRef))); assertFalse("dest node ref A2 has not been deleted", nodeService.exists(testNodeFactory.getMappedNodeRef(A2NodeRef))); assertTrue("dest node ref A3 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A3NodeRef))); assertFalse("dest node ref A4 has not been deleted", nodeService.exists(testNodeFactory.getMappedNodeRef(A4NodeRef))); assertFalse("dest node ref A5 has not been deleted", nodeService.exists(testNodeFactory.getMappedNodeRef(A5NodeRef))); } finally { endTransaction(); } startNewTransaction(); try { NodeRef archivedNode = new NodeRef(StoreRef.STORE_REF_ARCHIVE_SPACESSTORE, A2NodeRef.getId()); nodeService.restoreNode(archivedNode, A1NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A2")); } finally { endTransaction(); } startNewTransaction(); try { /** * Transfer Node A 1-5. * (So we are seeing what happens to node 2, 4, 5 on the target */ { TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(A1NodeRef); nodes.add(A2NodeRef); nodes.add(A3NodeRef); nodes.add(A4NodeRef); nodes.add(A5NodeRef); definition.setNodes(nodes); definition.setSync(true); transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source destNodeRef = testNodeFactory.getMappedNodeRef(A1NodeRef); assertFalse("unit test stuffed up - comparing with self", destNodeRef.equals(transferMe.getNodeRef())); assertTrue("dest node ref A1 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A1NodeRef))); assertTrue("dest node ref A2 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A2NodeRef))); assertTrue("dest node ref A3 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A3NodeRef))); assertTrue("dest node ref A4 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A4NodeRef))); assertTrue("dest node ref A5 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A5NodeRef))); } finally { endTransaction(); } } /** * Test the transfer method behaviour with respect to sync with (local) alien nodes. * * So we have Repository A transferring content and Repository B is the local repo that we * add and delete alien nodes. * * In general an alien node will prevent deletion of the parent folders * *
     * Tree of nodes
     * 
     *      A1
     *      |      |                      | 
     *      A2     A3 (Content Node)      B9 Alien Content Node
     *      |
     *   A4 A5 B10 (Alien Content Node)   A6
     *   |                                |
     *   A7 B11 (Alien Content Node)      A8 B12 B13 Alien Contact Nodes
     * 
* Test steps - *
    *
  1. add A1, A2, A3, A4, A5, A6, A7, A8 * transfer(sync)
  2. *
  3. add Alien node B9. A1 becomes Alien.
  4. *
  5. remove alien node B9. A1 becomes non Alien.
  6. *
  7. add Alien node B10. A1 and A2 become Alien
  8. *
  9. remove Alien node B10. A1 and A2 become non Alien
  10. *
  11. add B12 A6, A2, A1 becomes Alien
  12. *
  13. add B13 A6, A2, A1 remains Alien
  14. *
  15. remove B13 A6, A2, A1 remains Alien
  16. *
  17. remove B12 A6, A2, A1 becomes non Alien.
  18. *
  19. add B9 and B10 A1 and A2 become Alien
  20. *
  21. remove B10 A2 becomes non alien A1 remains alien.
  22. *
  23. Add B11, delete A2 * transfer sync
  24. * (A5, A6, A7 and A8 should be deleted A2 and A4 remain since they contain alien content.) *
* * TODO test move and alien * * TODO test restore and alien * */ public void testTransferInvadedByLocalAlienNodes() throws Exception { setDefaultRollback(false); String CONTENT_TITLE = "ContentTitle"; String CONTENT_TITLE_UPDATED = "ContentTitleUpdated"; Locale CONTENT_LOCALE = Locale.JAPAN; String CONTENT_STRING = "Hello"; /** * For unit test * - replace the HTTP transport with the in-process transport * - replace the node factory with one that will map node refs, paths etc. * * Mock the transfer service to be from Repo A */ TransferTransmitter transmitter = new UnitTestInProcessTransmitterImpl(receiver, contentService, transactionService); transferServiceImpl.setTransmitter(transmitter); UnitTestTransferManifestNodeFactory testNodeFactory = new UnitTestTransferManifestNodeFactory(this.transferManifestNodeFactory); transferServiceImpl.setTransferManifestNodeFactory(testNodeFactory); List> pathMap = testNodeFactory.getPathMap(); // Map company_home/guest_home to company_home so tranferred nodes and moved "up" one level. pathMap.add(new Pair(PathHelper.stringToPath(GUEST_HOME_XPATH_QUERY), PathHelper.stringToPath(COMPANY_HOME_XPATH_QUERY))); final String localRepositoryId = descriptorService.getCurrentRepositoryDescriptor().getId(); DescriptorService mockedDescriptorService = getMockDescriptorService(REPO_ID_A); transferServiceImpl.setDescriptorService(mockedDescriptorService); /** * Now go ahead and create our first transfer target */ String targetName = "testSyncWithAlienNodes"; TransferTarget transferMe; NodeRef A1NodeRef; NodeRef A2NodeRef; NodeRef A3NodeRef; NodeRef A4NodeRef; NodeRef A5NodeRef; NodeRef A6NodeRef; NodeRef A7NodeRef; NodeRef A8NodeRef; NodeRef B9NodeRef; NodeRef B10NodeRef; NodeRef B11NodeRef; NodeRef B12NodeRef; NodeRef B13NodeRef; NodeRef destNodeRef; startNewTransaction(); try { /** * Get guest home */ String guestHomeQuery = "/app:company_home/app:guest_home"; ResultSet guestHomeResult = searchService.query(StoreRef.STORE_REF_WORKSPACE_SPACESSTORE, SearchService.LANGUAGE_XPATH, guestHomeQuery); assertEquals("", 1, guestHomeResult.length()); NodeRef guestHome = guestHomeResult.getNodeRef(0); /** * Create a test nodes A1 through A8 that we will read and write */ { // Node A1 String name = GUID.generate(); ChildAssociationRef child = nodeService.createNode(guestHome, ContentModel.ASSOC_CONTAINS, QName.createQName(name), ContentModel.TYPE_FOLDER); A1NodeRef = child.getChildRef(); nodeService.setProperty(A1NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(A1NodeRef, ContentModel.PROP_NAME, name); } { // Node A2 ChildAssociationRef child = nodeService.createNode(A1NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A2"), ContentModel.TYPE_FOLDER); A2NodeRef = child.getChildRef(); nodeService.setProperty(A2NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(A2NodeRef, ContentModel.PROP_NAME, "A2"); } { // Node A3 ChildAssociationRef child = nodeService.createNode(A1NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A3"), ContentModel.TYPE_CONTENT); A3NodeRef = child.getChildRef(); nodeService.setProperty(A3NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(A3NodeRef, ContentModel.PROP_NAME, "A3"); ContentWriter writer = contentService.getWriter(A3NodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } { // Node A4 ChildAssociationRef child = nodeService.createNode(A2NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A4"), ContentModel.TYPE_FOLDER); A4NodeRef = child.getChildRef(); nodeService.setProperty(A4NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(A4NodeRef, ContentModel.PROP_NAME, "A4"); } { // Node A5 ChildAssociationRef child = nodeService.createNode(A2NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A5"), ContentModel.TYPE_CONTENT); A5NodeRef = child.getChildRef(); nodeService.setProperty(A5NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(A5NodeRef, ContentModel.PROP_NAME, "A5"); ContentWriter writer = contentService.getWriter(A5NodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } { // Node A6 ChildAssociationRef child = nodeService.createNode(A2NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A6"), ContentModel.TYPE_FOLDER); A6NodeRef = child.getChildRef(); nodeService.setProperty(A6NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(A6NodeRef, ContentModel.PROP_NAME, "A6"); } { // Node A7 ChildAssociationRef child = nodeService.createNode(A4NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A7"), ContentModel.TYPE_CONTENT); A7NodeRef = child.getChildRef(); nodeService.setProperty(A7NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(A7NodeRef, ContentModel.PROP_NAME, "A7"); ContentWriter writer = contentService.getWriter(A7NodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } { // Node A8 ChildAssociationRef child = nodeService.createNode(A6NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A8"), ContentModel.TYPE_CONTENT); A8NodeRef = child.getChildRef(); nodeService.setProperty(A8NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(A8NodeRef, ContentModel.PROP_NAME, "A8"); ContentWriter writer = contentService.getWriter(A8NodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } // Create the transfer target if it does not already exist if(!transferService.targetExists(targetName)) { transferMe = createTransferTarget(targetName); } else { transferMe = transferService.getTransferTarget(targetName); } } finally { endTransaction(); } /** * Step 1. add A1, A2, A3, A4, A5, A6, A7, A8 * transfer(sync) */ startNewTransaction(); try { /** * Transfer Nodes A1 through A8 */ { TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(A1NodeRef); nodes.add(A2NodeRef); nodes.add(A3NodeRef); nodes.add(A4NodeRef); nodes.add(A5NodeRef); nodes.add(A6NodeRef); nodes.add(A7NodeRef); nodes.add(A8NodeRef); definition.setNodes(nodes); definition.setSync(true); transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source NodeRef A1destNodeRef = testNodeFactory.getMappedNodeRef(A1NodeRef); assertFalse("unit test stuffed up - comparing with self", A1destNodeRef.equals(transferMe.getNodeRef())); assertTrue("dest node ref does not exist", nodeService.exists(A1destNodeRef)); assertEquals("title is wrong", (String)nodeService.getProperty(A1destNodeRef, ContentModel.PROP_TITLE), CONTENT_TITLE); assertEquals("type is wrong", nodeService.getType(A1NodeRef), nodeService.getType(A1destNodeRef)); assertFalse("A1 is alien", nodeService.hasAspect(A1destNodeRef, TransferModel.ASPECT_ALIEN)); // Check injected transferred aspect. assertNotNull("transferredAspect", (String)nodeService.getProperty(A1destNodeRef, TransferModel.PROP_REPOSITORY_ID)); } finally { endTransaction(); } /** * Step 2 add Alien node B9 child of A1(dest). A1(dest) becomes Alien because it contains an alien child. */ startNewTransaction(); try { destNodeRef = testNodeFactory.getMappedNodeRef(A1NodeRef); ChildAssociationRef child = nodeService.createNode(destNodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("B9"), ContentModel.TYPE_CONTENT); B9NodeRef = child.getChildRef(); nodeService.setProperty(B9NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(B9NodeRef, ContentModel.PROP_NAME, "B9"); ContentWriter writer = contentService.getWriter(B9NodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source NodeRef A1destNodeRef = testNodeFactory.getMappedNodeRef(A1NodeRef); assertTrue("dest node ref does not exist", nodeService.exists(A1destNodeRef)); // Check injected transferred aspect. assertTrue("node A1 is not alien aspect", (Boolean)nodeService.hasAspect(A1destNodeRef, TransferModel.ASPECT_ALIEN)); assertNotNull("repository id is null", (String)nodeService.getProperty(A1destNodeRef, TransferModel.PROP_REPOSITORY_ID)); assertNotNull("from repository id is null", (String)nodeService.getProperty(A1destNodeRef, TransferModel.PROP_FROM_REPOSITORY_ID)); assertTrue("node B9 is not alien", (Boolean)nodeService.hasAspect(B9NodeRef, TransferModel.ASPECT_ALIEN)); // Temp code List invaders = (List) nodeService.getProperty(A1destNodeRef, TransferModel.PROP_INVADED_BY); assertTrue("invaders contains local repository Id", invaders.contains(localRepositoryId)); assertFalse("invaders contains REPO_ID_A", invaders.contains(REPO_ID_A)); logger.debug("MER WOZ ERE" + invaders); } finally { endTransaction(); } /** * Step 3 remove alien node B9. A1 becomes non Alien. */ startNewTransaction(); try { logger.debug("delete node B9"); nodeService.deleteNode(B9NodeRef); } finally { endTransaction(); } startNewTransaction(); try { NodeRef A1destNodeRef = testNodeFactory.getMappedNodeRef(A1NodeRef); NodeRef A3destNodeRef = testNodeFactory.getMappedNodeRef(A3NodeRef); // Temp code List invaders = (List) nodeService.getProperty(A1destNodeRef, TransferModel.PROP_INVADED_BY); logger.debug("MER WOZ ERE AFTER B9 deleted" + invaders); assertTrue("dest node ref does not exist", nodeService.exists(A1destNodeRef)); assertFalse("node A1 is still alien", (Boolean)nodeService.hasAspect(A1destNodeRef, TransferModel.ASPECT_ALIEN)); assertFalse("node A3 is alien", (Boolean)nodeService.hasAspect(A3destNodeRef, TransferModel.ASPECT_ALIEN)); assertNotNull("repository id is null", (String)nodeService.getProperty(A1destNodeRef, TransferModel.PROP_REPOSITORY_ID)); assertNotNull("from repository id is null", (String)nodeService.getProperty(A1destNodeRef, TransferModel.PROP_FROM_REPOSITORY_ID)); } finally { endTransaction(); } /** * 4 add Alien node B10 child of A2. A1 and A2 become Alien */ startNewTransaction(); try { destNodeRef = testNodeFactory.getMappedNodeRef(A2NodeRef); ChildAssociationRef child = nodeService.createNode(destNodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("B10"), ContentModel.TYPE_CONTENT); B10NodeRef = child.getChildRef(); nodeService.setProperty(B10NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(B10NodeRef, ContentModel.PROP_NAME, "B10"); ContentWriter writer = contentService.getWriter(B10NodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source NodeRef A1destNodeRef = testNodeFactory.getMappedNodeRef(A1NodeRef); NodeRef A2destNodeRef = testNodeFactory.getMappedNodeRef(A2NodeRef); assertTrue("node A1 is not alien aspect", (Boolean)nodeService.hasAspect(A1destNodeRef, TransferModel.ASPECT_ALIEN)); assertTrue("node A2 is not alien aspect", (Boolean)nodeService.hasAspect(A2destNodeRef, TransferModel.ASPECT_ALIEN)); } finally { endTransaction(); } /** * 5 remove Alien node B10. A1 and A2 become non Alien */ startNewTransaction(); try { logger.debug("delete node B10"); nodeService.deleteNode(B10NodeRef); } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source NodeRef A1destNodeRef = testNodeFactory.getMappedNodeRef(A1NodeRef); NodeRef A2destNodeRef = testNodeFactory.getMappedNodeRef(A2NodeRef); assertFalse("node A1 is still alien aspect", (Boolean)nodeService.hasAspect(A1destNodeRef, TransferModel.ASPECT_ALIEN)); assertFalse("node A2 is still alien aspect", (Boolean)nodeService.hasAspect(A2destNodeRef, TransferModel.ASPECT_ALIEN)); } finally { endTransaction(); } /** * Step 6 * add B12 (child of A6) A6, A2, A1 becomes Alien */ startNewTransaction(); try { destNodeRef = testNodeFactory.getMappedNodeRef(A6NodeRef); ChildAssociationRef child = nodeService.createNode(destNodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("B12"), ContentModel.TYPE_CONTENT); B12NodeRef = child.getChildRef(); nodeService.setProperty(B12NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(B12NodeRef, ContentModel.PROP_NAME, "B12"); ContentWriter writer = contentService.getWriter(B12NodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source NodeRef A1destNodeRef = testNodeFactory.getMappedNodeRef(A1NodeRef); NodeRef A2destNodeRef = testNodeFactory.getMappedNodeRef(A2NodeRef); NodeRef A6destNodeRef = testNodeFactory.getMappedNodeRef(A6NodeRef); assertTrue("node A1 is not alien", (Boolean)nodeService.hasAspect(A1destNodeRef, TransferModel.ASPECT_ALIEN)); assertTrue("node A2 is not alien", (Boolean)nodeService.hasAspect(A2destNodeRef, TransferModel.ASPECT_ALIEN)); assertTrue("node A6 is not alien", (Boolean)nodeService.hasAspect(A6destNodeRef, TransferModel.ASPECT_ALIEN)); } finally { endTransaction(); } /** * Step 7 * add B13 A6, A2, A1 remains Alien */ startNewTransaction(); try { destNodeRef = testNodeFactory.getMappedNodeRef(A6NodeRef); ChildAssociationRef child = nodeService.createNode(destNodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("B13"), ContentModel.TYPE_CONTENT); B13NodeRef = child.getChildRef(); nodeService.setProperty(B13NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(B13NodeRef, ContentModel.PROP_NAME, "B13"); ContentWriter writer = contentService.getWriter(B13NodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source NodeRef A1destNodeRef = testNodeFactory.getMappedNodeRef(A1NodeRef); NodeRef A2destNodeRef = testNodeFactory.getMappedNodeRef(A2NodeRef); NodeRef A6destNodeRef = testNodeFactory.getMappedNodeRef(A6NodeRef); assertTrue("node A1 is not alien", (Boolean)nodeService.hasAspect(A1destNodeRef, TransferModel.ASPECT_ALIEN)); assertTrue("node A2 is not alien", (Boolean)nodeService.hasAspect(A2destNodeRef, TransferModel.ASPECT_ALIEN)); assertTrue("node A6 is not alien", (Boolean)nodeService.hasAspect(A6destNodeRef, TransferModel.ASPECT_ALIEN)); } finally { endTransaction(); } /** * Step 8 remove B13 A6, A2, A1 remains Alien Due to B12 */ startNewTransaction(); try { nodeService.deleteNode(B13NodeRef); } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source NodeRef A1destNodeRef = testNodeFactory.getMappedNodeRef(A1NodeRef); NodeRef A2destNodeRef = testNodeFactory.getMappedNodeRef(A2NodeRef); NodeRef A6destNodeRef = testNodeFactory.getMappedNodeRef(A6NodeRef); assertTrue("node A1 is not alien", (Boolean)nodeService.hasAspect(A1destNodeRef, TransferModel.ASPECT_ALIEN)); assertTrue("node A2 is not alien", (Boolean)nodeService.hasAspect(A2destNodeRef, TransferModel.ASPECT_ALIEN)); assertTrue("node A6 is not alien", (Boolean)nodeService.hasAspect(A6destNodeRef, TransferModel.ASPECT_ALIEN)); } finally { endTransaction(); } /** * Step 9 remove B12 A6, A2, A1 becomes non Alien. */ startNewTransaction(); try { nodeService.deleteNode(B12NodeRef); } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source NodeRef A1destNodeRef = testNodeFactory.getMappedNodeRef(A1NodeRef); NodeRef A2destNodeRef = testNodeFactory.getMappedNodeRef(A2NodeRef); NodeRef A6destNodeRef = testNodeFactory.getMappedNodeRef(A6NodeRef); assertFalse("node A1 is still alien", (Boolean)nodeService.hasAspect(A1destNodeRef, TransferModel.ASPECT_ALIEN)); assertFalse("node A2 is still alien", (Boolean)nodeService.hasAspect(A2destNodeRef, TransferModel.ASPECT_ALIEN)); assertFalse("node A6 is still alien", (Boolean)nodeService.hasAspect(A6destNodeRef, TransferModel.ASPECT_ALIEN)); } finally { endTransaction(); } /** * Step 10 add B9 and B10 A1 and A2 become Alien */ startNewTransaction(); try { destNodeRef = testNodeFactory.getMappedNodeRef(A1NodeRef); ChildAssociationRef child = nodeService.createNode(destNodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("B9"), ContentModel.TYPE_CONTENT); B9NodeRef = child.getChildRef(); nodeService.setProperty(B9NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(B9NodeRef, ContentModel.PROP_NAME, "B9"); ContentWriter writer = contentService.getWriter(B9NodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); destNodeRef = testNodeFactory.getMappedNodeRef(A2NodeRef); child = nodeService.createNode(destNodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("B10"), ContentModel.TYPE_CONTENT); B10NodeRef = child.getChildRef(); nodeService.setProperty(B10NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(B10NodeRef, ContentModel.PROP_NAME, "B10"); writer = contentService.getWriter(B10NodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source NodeRef A1destNodeRef = testNodeFactory.getMappedNodeRef(A1NodeRef); NodeRef A2destNodeRef = testNodeFactory.getMappedNodeRef(A2NodeRef); assertTrue("node A1 is not alien", (Boolean)nodeService.hasAspect(A1destNodeRef, TransferModel.ASPECT_ALIEN)); assertTrue("node A2 is not alien", (Boolean)nodeService.hasAspect(A2destNodeRef, TransferModel.ASPECT_ALIEN)); } finally { endTransaction(); } /** * Step 11 remove B10 A2 becomes non alien A1 remains alien. */ startNewTransaction(); try { nodeService.deleteNode(B10NodeRef); } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source NodeRef A1destNodeRef = testNodeFactory.getMappedNodeRef(A1NodeRef); NodeRef A2destNodeRef = testNodeFactory.getMappedNodeRef(A2NodeRef); // BUGBUG assertTrue("node A1 is still alien", (Boolean)nodeService.hasAspect(A1destNodeRef, TransferModel.ASPECT_ALIEN)); assertFalse("node A2 is still alien", (Boolean)nodeService.hasAspect(A2destNodeRef, TransferModel.ASPECT_ALIEN)); } finally { endTransaction(); } /** * 12 Add Alien node B11. * delete A2 (will cascade delete A4, A5, A6, A7, A8 * transfer sync * (A5, A6, A7, A8 and should be deleted A2 and A4 remain since they contain alien content.) */ logger.debug("Step 12 Add Node B11, Delete A2 and sync"); startNewTransaction(); try { destNodeRef = testNodeFactory.getMappedNodeRef(A4NodeRef); ChildAssociationRef child = nodeService.createNode(destNodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("B11"), ContentModel.TYPE_CONTENT); B11NodeRef = child.getChildRef(); nodeService.setProperty(B11NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(B11NodeRef, ContentModel.PROP_NAME, "B11"); ContentWriter writer = contentService.getWriter(B11NodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); nodeService.deleteNode(A2NodeRef); } finally { endTransaction(); } startNewTransaction(); try { // Now validate A1, A2 and A4 are alien NodeRef A1destNodeRef = testNodeFactory.getMappedNodeRef(A1NodeRef); NodeRef A2destNodeRef = testNodeFactory.getMappedNodeRef(A2NodeRef); NodeRef A4destNodeRef = testNodeFactory.getMappedNodeRef(A4NodeRef); assertTrue("node A1 is not alien", (Boolean)nodeService.hasAspect(A1destNodeRef, TransferModel.ASPECT_ALIEN)); assertTrue("node A2 is not alien", (Boolean)nodeService.hasAspect(A2destNodeRef, TransferModel.ASPECT_ALIEN)); assertTrue("node A4 is not alien", (Boolean)nodeService.hasAspect(A4destNodeRef, TransferModel.ASPECT_ALIEN)); assertFalse("test error: node A2 not deleted", nodeService.exists(A2NodeRef)); assertFalse("test error: node A4 not deleted", nodeService.exists(A4NodeRef)); assertFalse("test error: node A5 not deleted", nodeService.exists(A5NodeRef)); assertFalse("test error: node A6 not deleted", nodeService.exists(A6NodeRef)); assertFalse("test error: node A7 not deleted", nodeService.exists(A7NodeRef)); assertFalse("test error: node A8 not deleted", nodeService.exists(A8NodeRef)); /** * Transfer Nodes A1 through A8 */ { TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(A1NodeRef); nodes.add(A3NodeRef); definition.setNodes(nodes); definition.setSync(true); transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists and has similar properties to the source NodeRef A1destNodeRef = testNodeFactory.getMappedNodeRef(A1NodeRef); NodeRef A2destNodeRef = testNodeFactory.getMappedNodeRef(A2NodeRef); NodeRef A3destNodeRef = testNodeFactory.getMappedNodeRef(A3NodeRef); NodeRef A4destNodeRef = testNodeFactory.getMappedNodeRef(A4NodeRef); NodeRef A5destNodeRef = testNodeFactory.getMappedNodeRef(A5NodeRef); NodeRef A6destNodeRef = testNodeFactory.getMappedNodeRef(A6NodeRef); NodeRef A7destNodeRef = testNodeFactory.getMappedNodeRef(A7NodeRef); NodeRef A8destNodeRef = testNodeFactory.getMappedNodeRef(A8NodeRef); assertTrue("node A1 not alien", (Boolean)nodeService.hasAspect(A1destNodeRef, TransferModel.ASPECT_ALIEN)); assertTrue("node A2 not alien", (Boolean)nodeService.hasAspect(A2destNodeRef, TransferModel.ASPECT_ALIEN)); assertTrue("node A4 not alien", (Boolean)nodeService.hasAspect(A4destNodeRef, TransferModel.ASPECT_ALIEN)); assertTrue("node B11 does not exist", nodeService.exists(B11NodeRef)); assertTrue("node A3 deleted", nodeService.exists(A3destNodeRef)); assertFalse("node A5 not deleted", nodeService.exists(A5destNodeRef)); assertFalse("node A6 not deleted", nodeService.exists(A6destNodeRef)); assertFalse("node A7 not deleted", nodeService.exists(A7destNodeRef)); assertFalse("node A8 not deleted", nodeService.exists(A8destNodeRef)); } finally { endTransaction(); } } /** * Test the transfer method with regard to permissions on a node. *

* Step 1: * Create a node with a single permission * Inherit:false * Read, Admin, Allow * Transfer *

* Step 2: * Update it to have several permissions * Inherit:false * Read, Everyone, DENY * Read, Admin, Allow *

* Step 3: * Remove a permission * Inherit:false * Read, Admin, Allow *

* Step 4: * Revert to inherit all permissions * Inherit:true *

* This is a unit test so it does some shenanigans to send to the same instance of alfresco. */ public void testTransferWithPermissions() throws Exception { setDefaultRollback(false); String CONTENT_TITLE = "ContentTitle"; String CONTENT_TITLE_UPDATED = "ContentTitleUpdated"; Locale CONTENT_LOCALE = Locale.GERMAN; String CONTENT_STRING = "Hello"; /** * For unit test * - replace the HTTP transport with the in-process transport * - replace the node factory with one that will map node refs, paths etc. */ TransferTransmitter transmitter = new UnitTestInProcessTransmitterImpl(receiver, contentService, transactionService); transferServiceImpl.setTransmitter(transmitter); UnitTestTransferManifestNodeFactory testNodeFactory = new UnitTestTransferManifestNodeFactory(this.transferManifestNodeFactory); transferServiceImpl.setTransferManifestNodeFactory(testNodeFactory); List> pathMap = testNodeFactory.getPathMap(); // Map company_home/guest_home to company_home so tranferred nodes and moved "up" one level. pathMap.add(new Pair(PathHelper.stringToPath(GUEST_HOME_XPATH_QUERY), PathHelper.stringToPath(COMPANY_HOME_XPATH_QUERY))); DescriptorService mockedDescriptorService = getMockDescriptorService(REPO_ID_A); transferServiceImpl.setDescriptorService(mockedDescriptorService); /** * Now go ahead and create our transfer target */ String targetName = "testTransferWithPermissions"; TransferTarget transferMe; NodeRef contentNodeRef; NodeRef destNodeRef; startNewTransaction(); try { /** * Get guest home */ String guestHomeQuery = "/app:company_home/app:guest_home"; ResultSet guestHomeResult = searchService.query(StoreRef.STORE_REF_WORKSPACE_SPACESSTORE, SearchService.LANGUAGE_XPATH, guestHomeQuery); assertEquals("", 1, guestHomeResult.length()); NodeRef guestHome = guestHomeResult.getNodeRef(0); /** * Create a test node that we will read and write */ String name = GUID.generate(); ChildAssociationRef child = nodeService.createNode(guestHome, ContentModel.ASSOC_CONTAINS, QName.createQName(name), ContentModel.TYPE_CONTENT); contentNodeRef = child.getChildRef(); nodeService.setProperty(contentNodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(contentNodeRef, ContentModel.PROP_NAME, name); ContentWriter writer = contentService.getWriter(contentNodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); permissionService.setInheritParentPermissions(contentNodeRef, false); permissionService.setPermission(contentNodeRef, "admin", PermissionService.READ, true); if(!transferService.targetExists(targetName)) { transferMe = createTransferTarget(targetName); } else { transferMe = transferService.getTransferTarget(targetName); } } finally { endTransaction(); } /** * Step 1 */ logger.debug("First transfer - create new node with inheritParent permission off"); startNewTransaction(); try { /** * Transfer our transfer target node */ { TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(contentNodeRef); definition.setNodes(nodes); transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists with the correct permissions destNodeRef = testNodeFactory.getMappedNodeRef(contentNodeRef); assertFalse("unit test stuffed up - comparing with self", destNodeRef.equals(transferMe.getNodeRef())); assertTrue("dest node ref does not exist", nodeService.exists(destNodeRef)); assertEquals("title is wrong", (String)nodeService.getProperty(destNodeRef, ContentModel.PROP_TITLE), CONTENT_TITLE); assertEquals("type is wrong", nodeService.getType(contentNodeRef), nodeService.getType(destNodeRef)); // Check ACL of destination node boolean srcInherit = permissionService.getInheritParentPermissions(contentNodeRef); Set srcPerm = permissionService.getAllSetPermissions(contentNodeRef); boolean destInherit = permissionService.getInheritParentPermissions(destNodeRef); Set destPerm = permissionService.getAllSetPermissions(destNodeRef); assertFalse("inherit parent permissions (src) flag is incorrect", srcInherit); assertFalse("inherit parent permissions (dest) flag is incorrect", destInherit); // Check destination has the source's permissions for (AccessPermission p : srcPerm) { logger.debug("checking permission :" + p); assertTrue("permission is missing", destPerm.contains(p)); } } finally { endTransaction(); } /** * Step 2 * Update it to have several permissions * Inherit:false * Read, Everyone, DENY * Read, Admin, Allow */ startNewTransaction(); try { permissionService.setPermission(contentNodeRef, "EVERYONE", PermissionService.READ, false); permissionService.setPermission(contentNodeRef, "admin", PermissionService.FULL_CONTROL, true); } finally { endTransaction(); } startNewTransaction(); try { /** * Transfer our transfer target node */ { TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(contentNodeRef); definition.setNodes(nodes); transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists with the correct permissions destNodeRef = testNodeFactory.getMappedNodeRef(contentNodeRef); // Check ACL of destination node boolean srcInherit = permissionService.getInheritParentPermissions(contentNodeRef); Set srcPerm = permissionService.getAllSetPermissions(contentNodeRef); boolean destInherit = permissionService.getInheritParentPermissions(destNodeRef); Set destPerm = permissionService.getAllSetPermissions(destNodeRef); assertFalse("inherit parent permissions (src) flag is incorrect", srcInherit); assertFalse("inherit parent permissions (dest) flag is incorrect", destInherit); // Check destination has the source's permissions for (AccessPermission p : srcPerm) { logger.debug("checking permission :" + p); assertTrue("Step2, permission is missing", destPerm.contains(p)); } } finally { endTransaction(); } /** * Step 3 Remove a permission */ startNewTransaction(); try { permissionService.deletePermission(contentNodeRef, "admin", PermissionService.FULL_CONTROL); } finally { endTransaction(); } startNewTransaction(); try { /** * Transfer our transfer target node */ { TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(contentNodeRef); definition.setNodes(nodes); transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists with the correct permissions destNodeRef = testNodeFactory.getMappedNodeRef(contentNodeRef); // Check ACL of destination node boolean srcInherit = permissionService.getInheritParentPermissions(contentNodeRef); Set srcPerm = permissionService.getAllSetPermissions(contentNodeRef); boolean destInherit = permissionService.getInheritParentPermissions(destNodeRef); Set destPerm = permissionService.getAllSetPermissions(destNodeRef); assertFalse("inherit parent permissions (src) flag is incorrect", srcInherit); assertFalse("inherit parent permissions (dest) flag is incorrect", destInherit); // Check destination has the source's permissions for (AccessPermission p : srcPerm) { logger.debug("checking permission :" + p); assertTrue("permission is missing", destPerm.contains(p)); } } finally { endTransaction(); } /** * Step 4 * Revert to inherit all permissions */ startNewTransaction(); try { permissionService.setInheritParentPermissions(contentNodeRef, true); } finally { endTransaction(); } startNewTransaction(); try { /** * Transfer our transfer target node */ { TransferDefinition definition = new TransferDefinition(); Setnodes = new HashSet(); nodes.add(contentNodeRef); definition.setNodes(nodes); transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { // Now validate that the target node exists with the correct permissions destNodeRef = testNodeFactory.getMappedNodeRef(contentNodeRef); assertFalse("unit test stuffed up - comparing with self", destNodeRef.equals(transferMe.getNodeRef())); assertTrue("dest node ref does not exist", nodeService.exists(destNodeRef)); assertEquals("title is wrong", (String)nodeService.getProperty(destNodeRef, ContentModel.PROP_TITLE), CONTENT_TITLE); assertEquals("type is wrong", nodeService.getType(contentNodeRef), nodeService.getType(destNodeRef)); // Check ACL of destination node boolean srcInherit = permissionService.getInheritParentPermissions(contentNodeRef); Set srcPerm = permissionService.getAllSetPermissions(contentNodeRef); boolean destInherit = permissionService.getInheritParentPermissions(destNodeRef); Set destPerm = permissionService.getAllSetPermissions(destNodeRef); assertTrue("inherit parent permissions (src) flag is incorrect", srcInherit); assertTrue("inherit parent permissions (dest) flag is incorrect", destInherit); // Check destination has the source's permissions for (AccessPermission p : srcPerm) { if(p.isSetDirectly()) { logger.debug("checking permission :" + p); assertTrue("permission is missing:" + p, destPerm.contains(p)); } } } finally { endTransaction(); } } /** * Transfer with read only flag * * Node tree for this test *

 
     *           A (Folder)
     *   |                 | 
     *   B (Content)   C (Folder)
     *                     |
     *                     D (Content)
     * 
* Step 1 * transfer Nodes ABCD with read only flag set - content should all be locked on destination *

* Step 2 * lock B (Content node) as user fred * transfer (read only) - destination lock should change to Admin *

* Step 3 * lock C (Folder) as user fred * transfer (read only) - destination lock should change to Admin *

* Step 4 * transfer without read only flag - locks should revert from Admin to Fred. *

* Step 5 * remove locks on A and B - transfer without read only flag - content should all be unlocked. */ public void testReadOnlyFlag() throws Exception { setDefaultRollback(false); String CONTENT_TITLE = "ContentTitle"; String CONTENT_TITLE_UPDATED = "ContentTitleUpdated"; String CONTENT_NAME = "Demo Node 1"; Locale CONTENT_LOCALE = Locale.GERMAN; String CONTENT_STRING = "The quick brown fox"; Setnodes = new HashSet(); String USER_ONE = "TransferServiceImplTest"; String PASSWORD = "Password"; String targetName = "testReadOnlyFlag"; NodeRef nodeA; NodeRef nodeB; NodeRef nodeC; NodeRef nodeD; ChildAssociationRef child; /** * For unit test * - replace the HTTP transport with the in-process transport * - replace the node factory with one that will map node refs, paths etc. */ TransferTransmitter transmitter = new UnitTestInProcessTransmitterImpl(this.receiver, this.contentService, transactionService); transferServiceImpl.setTransmitter(transmitter); UnitTestTransferManifestNodeFactory testNodeFactory = new UnitTestTransferManifestNodeFactory(this.transferManifestNodeFactory); transferServiceImpl.setTransferManifestNodeFactory(testNodeFactory); List> pathMap = testNodeFactory.getPathMap(); // Map company_home/guest_home to company_home so tranferred nodes and moved "up" one level. pathMap.add(new Pair(PathHelper.stringToPath(GUEST_HOME_XPATH_QUERY), PathHelper.stringToPath(COMPANY_HOME_XPATH_QUERY))); DescriptorService mockedDescriptorService = getMockDescriptorService(REPO_ID_A); transferServiceImpl.setDescriptorService(mockedDescriptorService); TransferTarget transferMe; startNewTransaction(); try { /** * Get guest home */ String guestHomeQuery = "/app:company_home/app:guest_home"; ResultSet guestHomeResult = searchService.query(StoreRef.STORE_REF_WORKSPACE_SPACESSTORE, SearchService.LANGUAGE_XPATH, guestHomeQuery); assertEquals("", 1, guestHomeResult.length()); NodeRef guestHome = guestHomeResult.getNodeRef(0); /** * Create a test node that we will read and write */ String guid = GUID.generate(); child = nodeService.createNode(guestHome, ContentModel.ASSOC_CONTAINS, QName.createQName(guid), ContentModel.TYPE_FOLDER); nodeA = child.getChildRef(); nodeService.setProperty(nodeA , ContentModel.PROP_TITLE, guid); nodeService.setProperty(nodeA , ContentModel.PROP_NAME, guid); nodes.add(nodeA); child = nodeService.createNode(nodeA, ContentModel.ASSOC_CONTAINS, QName.createQName("testNodeB"), ContentModel.TYPE_CONTENT); nodeB = child.getChildRef(); nodeService.setProperty(nodeB , ContentModel.PROP_TITLE, CONTENT_TITLE + "B"); nodeService.setProperty(nodeB , ContentModel.PROP_NAME, "DemoNodeB"); { ContentWriter writer = contentService.getWriter(nodeB , ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); nodes.add(nodeB); } child = nodeService.createNode(nodeA, ContentModel.ASSOC_CONTAINS, QName.createQName(NamespaceService.CONTENT_MODEL_1_0_URI,"testNodeC"), ContentModel.TYPE_FOLDER); nodeC = child.getChildRef(); nodeService.setProperty(nodeC , ContentModel.PROP_TITLE, "TestNodeC"); nodeService.setProperty(nodeC , ContentModel.PROP_NAME, "TestNodeC"); nodes.add(nodeC); child = nodeService.createNode(nodeC, ContentModel.ASSOC_CONTAINS, QName.createQName("testNodeD"), ContentModel.TYPE_CONTENT); nodeD = child.getChildRef(); nodeService.setProperty(nodeD , ContentModel.PROP_TITLE, CONTENT_TITLE + "D"); nodeService.setProperty(nodeD , ContentModel.PROP_NAME, "DemoNodeD"); { ContentWriter writer = contentService.getWriter(nodeD , ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); nodes.add(nodeD); } // Create users createUser(USER_ONE, PASSWORD); /** * Now go ahead and create our first transfer target */ if(!transferService.targetExists(targetName)) { transferMe = createTransferTarget(targetName); } else { transferMe = transferService.getTransferTarget(targetName); } } finally { endTransaction(); } /** * Step 1. * transfer Nodes ABCD with read only flag set - content should all be locked on destination */ logger.debug("transfer read only - step 1"); startNewTransaction(); try { /** * Transfer our transfer target nodes */ { TransferDefinition definition = new TransferDefinition(); definition.setNodes(nodes); definition.setReadOnly(true); transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { // Check destination nodes are locked. assertTrue("dest node A does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(nodeA))); assertTrue("dest node B does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(nodeB))); assertTrue("dest node C does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(nodeC))); assertTrue("dest node D does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(nodeD))); assertTrue("dest node A not locked", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(nodeA), ContentModel.ASPECT_LOCKABLE)); assertTrue("dest node B not locked", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(nodeB), ContentModel.ASPECT_LOCKABLE)); assertTrue("dest node C not locked", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(nodeC), ContentModel.ASPECT_LOCKABLE)); assertTrue("dest node D not locked", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(nodeD), ContentModel.ASPECT_LOCKABLE)); } finally { endTransaction(); } /** * Step 2 * lock B (Content node) as user ONE * transfer (read only) - destination lock should change user to "Admin" */ startNewTransaction(); try { AuthenticationUtil.pushAuthentication(); AuthenticationUtil.setFullyAuthenticatedUser(USER_ONE); lockService.lock(nodeB, LockType.READ_ONLY_LOCK); } finally { assertEquals("test error: dest node B lock ownership", nodeService.getProperty(nodeB, ContentModel.PROP_LOCK_OWNER), USER_ONE); AuthenticationUtil.popAuthentication(); endTransaction(); } logger.debug("transfer read only - step 2"); startNewTransaction(); try { /** * Transfer our transfer target nodes */ { TransferDefinition definition = new TransferDefinition(); definition.setNodes(nodes); definition.setReadOnly(true); transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { // Check destination nodes are locked. assertTrue("dest node A does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(nodeA))); assertTrue("dest node B does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(nodeB))); assertTrue("dest node C does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(nodeC))); assertTrue("dest node D does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(nodeD))); assertTrue("dest node A not locked", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(nodeA), ContentModel.ASPECT_LOCKABLE)); assertTrue("dest node B not locked", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(nodeB), ContentModel.ASPECT_LOCKABLE)); assertTrue("dest node C not locked", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(nodeC), ContentModel.ASPECT_LOCKABLE)); assertTrue("dest node D not locked", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(nodeD), ContentModel.ASPECT_LOCKABLE)); // check that the lock owner is no longer USER_ONE assertTrue("lock owner not changed", !USER_ONE.equalsIgnoreCase((String)nodeService.getProperty(testNodeFactory.getMappedNodeRef(nodeB), ContentModel.PROP_LOCK_OWNER))); } finally { endTransaction(); } /** * Step 3 * lock C (Folder node) as user ONE * transfer (read only) - destination lock should change to Admin */ startNewTransaction(); try { AuthenticationUtil.pushAuthentication(); AuthenticationUtil.setFullyAuthenticatedUser(USER_ONE); lockService.lock(nodeC, LockType.READ_ONLY_LOCK); } finally { assertEquals("test error: dest node C lock ownership", nodeService.getProperty(nodeC, ContentModel.PROP_LOCK_OWNER), USER_ONE); AuthenticationUtil.popAuthentication(); endTransaction(); } logger.debug("transfer read only - step 3"); startNewTransaction(); try { /** * Transfer our transfer target nodes */ { TransferDefinition definition = new TransferDefinition(); definition.setNodes(nodes); definition.setReadOnly(true); transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { // Check destination nodes are locked. assertTrue("dest node A does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(nodeA))); assertTrue("dest node B does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(nodeB))); assertTrue("dest node C does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(nodeC))); assertTrue("dest node D does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(nodeD))); assertTrue("dest node A not locked", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(nodeA), ContentModel.ASPECT_LOCKABLE)); assertTrue("dest node B not locked", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(nodeB), ContentModel.ASPECT_LOCKABLE)); assertTrue("dest node C not locked", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(nodeC), ContentModel.ASPECT_LOCKABLE)); assertTrue("dest node D not locked", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(nodeD), ContentModel.ASPECT_LOCKABLE)); // check that the lock owner is no longer USER_ONE for content node B and folder node C assertTrue("lock owner not changed", !USER_ONE.equalsIgnoreCase((String)nodeService.getProperty(testNodeFactory.getMappedNodeRef(nodeB), ContentModel.PROP_LOCK_OWNER))); assertTrue("lock owner not changed", !USER_ONE.equalsIgnoreCase((String)nodeService.getProperty(testNodeFactory.getMappedNodeRef(nodeC), ContentModel.PROP_LOCK_OWNER))); } finally { endTransaction(); } /** * Step 4 * transfer without read only flag - locks should revert from Admin to USER_ONE. */ logger.debug("transfer read only - step 4"); startNewTransaction(); try { /** * Transfer our transfer target nodes */ { TransferDefinition definition = new TransferDefinition(); definition.setNodes(nodes); definition.setReadOnly(false); // turn off read-only transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { // Check destination nodes are not locked. assertTrue("dest node A does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(nodeA))); assertTrue("dest node B does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(nodeB))); assertTrue("dest node C does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(nodeC))); assertTrue("dest node D does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(nodeD))); assertFalse("dest node A not locked", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(nodeA), ContentModel.ASPECT_LOCKABLE)); assertTrue("dest node B not locked", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(nodeB), ContentModel.ASPECT_LOCKABLE)); assertTrue("dest node C not locked", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(nodeC), ContentModel.ASPECT_LOCKABLE)); assertFalse("dest node D not locked", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(nodeD), ContentModel.ASPECT_LOCKABLE)); assertEquals("dest node B lock ownership", nodeService.getProperty(testNodeFactory.getMappedNodeRef(nodeB), ContentModel.PROP_LOCK_OWNER), USER_ONE); assertEquals("dest node C lock ownership", nodeService.getProperty(testNodeFactory.getMappedNodeRef(nodeC), ContentModel.PROP_LOCK_OWNER), USER_ONE); } finally { endTransaction(); } /** * Step 5 * remove locks on A and B - transfer without read only flag - content should all be unlocked. */ logger.debug("transfer read only - step 5"); startNewTransaction(); try { lockService.unlock(nodeB); lockService.unlock(nodeC); } finally { endTransaction(); } startNewTransaction(); try { /** * Transfer our transfer target nodes */ { TransferDefinition definition = new TransferDefinition(); definition.setNodes(nodes); definition.setReadOnly(false); // turn off read-only transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { // Check destination nodes are not locked. assertTrue("dest node A does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(nodeA))); assertTrue("dest node B does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(nodeB))); assertTrue("dest node C does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(nodeC))); assertTrue("dest node D does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(nodeD))); assertFalse("test fail: dest node B is still locked", nodeService.hasAspect(nodeB, ContentModel.ASPECT_LOCKABLE)); assertFalse("test fail: dest node C is still locked", nodeService.hasAspect(nodeC, ContentModel.ASPECT_LOCKABLE)); assertFalse("dest node A not locked", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(nodeA), ContentModel.ASPECT_LOCKABLE)); assertFalse("dest node B not locked", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(nodeB), ContentModel.ASPECT_LOCKABLE)); assertFalse("dest node C not locked", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(nodeC), ContentModel.ASPECT_LOCKABLE)); assertFalse("dest node D not locked", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(nodeD), ContentModel.ASPECT_LOCKABLE)); } finally { endTransaction(); } } // end test read only flag /** * Transfer sync from multiple repos. * * This is a unit test so does lots of shenanigans to fake transfer from three repositories on a single repo. * * Multi-repo sync depends upon the following pieces of functionality * a) transferred nodes are tagged with a trx:transferred aspect containing the originating repository * id and the from repository id * b) to support hub and spoke - when syncing don't imply delete nodes that are not "from" the transferring system * * * Tree of nodes * A1 * | | * A2 A3 (Content Node) B6 (Content Node) * | * A4 A5 B7 * * Step 1 * Hub and Spoke Sync * create Tree A1...A5 * transfer (sync) * check the transfered aspects on destination * create node B6. Fake its transfered aspect to be from Repo B. * transfer (sync) * * Step 2 * Chain Sync * Create Node A7 "from repo B". * Change Nodes A1 ... A5 source to be received "from repo B" * transfer * * //TO BE INVESTIGATED - Not yet implemented * //c) when syncing don't accept updates to nodes that are not "from" the transferring system. * //Step 3. * //Multiple repo update of the same node * //Change the transferred aspect for A5 destination to be "from" repo B * //Update A5 * //Transfer (sync) * // * //Step 4. * //Change the transferred aspect for A5 destination to be "from" repo B * //Update A5 * //Transfer A5 (normal) - should update */ public void testTwoRepoSync() throws Exception { /** * Step 1 * create Tree A1...A6 * transfer (sync) * check the transfered aspect * create node B6. Fake its transfered aspect to be from Repo B, Non Alien. * transfer (sync) */ setDefaultRollback(false); String CONTENT_TITLE = "ContentTitle"; String CONTENT_TITLE_UPDATED = "ContentTitleUpdated"; Locale CONTENT_LOCALE = Locale.GERMAN; String CONTENT_STRING = "Hello"; /** * For unit test * - replace the HTTP transport with the in-process transport * - replace the node factory with one that will map node refs, paths etc. */ TransferTransmitter transmitter = new UnitTestInProcessTransmitterImpl(receiver, contentService, transactionService); transferServiceImpl.setTransmitter(transmitter); UnitTestTransferManifestNodeFactory testNodeFactory = new UnitTestTransferManifestNodeFactory(this.transferManifestNodeFactory); transferServiceImpl.setTransferManifestNodeFactory(testNodeFactory); List> pathMap = testNodeFactory.getPathMap(); // Map company_home/guest_home to company_home so tranferred nodes and moved "up" one level. pathMap.add(new Pair(PathHelper.stringToPath(GUEST_HOME_XPATH_QUERY), PathHelper.stringToPath(COMPANY_HOME_XPATH_QUERY))); DescriptorService mockedDescriptorService = getMockDescriptorService(REPO_ID_A); transferServiceImpl.setDescriptorService(mockedDescriptorService); String repositoryId = REPO_ID_A; /** * Now go ahead and create our first transfer target */ String targetName = "testTransferSyncNodes"; TransferTarget transferMe; NodeRef A1NodeRef; NodeRef A2NodeRef; NodeRef A3NodeRef; NodeRef A4NodeRef; NodeRef A5NodeRef; NodeRef B6NodeRef; NodeRef A7NodeRef; startNewTransaction(); try { /** * Get guest home */ String guestHomeQuery = "/app:company_home/app:guest_home"; ResultSet guestHomeResult = searchService.query(StoreRef.STORE_REF_WORKSPACE_SPACESSTORE, SearchService.LANGUAGE_XPATH, guestHomeQuery); assertEquals("", 1, guestHomeResult.length()); NodeRef guestHome = guestHomeResult.getNodeRef(0); /** * Create a test nodes A1 through A5 that we will read and write */ { // Node A1 String name = GUID.generate(); ChildAssociationRef child = nodeService.createNode(guestHome, ContentModel.ASSOC_CONTAINS, QName.createQName(name), ContentModel.TYPE_FOLDER); A1NodeRef = child.getChildRef(); nodeService.setProperty(A1NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(A1NodeRef, ContentModel.PROP_NAME, name); } { // Node A2 ChildAssociationRef child = nodeService.createNode(A1NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A2"), ContentModel.TYPE_FOLDER); A2NodeRef = child.getChildRef(); nodeService.setProperty(A2NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(A2NodeRef, ContentModel.PROP_NAME, "A2"); } { // Node A3 ChildAssociationRef child = nodeService.createNode(A1NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A3"), ContentModel.TYPE_CONTENT); A3NodeRef = child.getChildRef(); nodeService.setProperty(A3NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(A3NodeRef, ContentModel.PROP_NAME, "A3"); ContentWriter writer = contentService.getWriter(A3NodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } { // Node A4 ChildAssociationRef child = nodeService.createNode(A2NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A4"), ContentModel.TYPE_CONTENT); A4NodeRef = child.getChildRef(); nodeService.setProperty(A4NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(A4NodeRef, ContentModel.PROP_NAME, "A4"); ContentWriter writer = contentService.getWriter(A4NodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } { // Node A5 ChildAssociationRef child = nodeService.createNode(A2NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A5"), ContentModel.TYPE_CONTENT); A5NodeRef = child.getChildRef(); nodeService.setProperty(A5NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(A5NodeRef, ContentModel.PROP_NAME, "A5"); ContentWriter writer = contentService.getWriter(A5NodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } // Create the transfer target if it does not already exist if(!transferService.targetExists(targetName)) { transferMe = createTransferTarget(targetName); } else { transferMe = transferService.getTransferTarget(targetName); } } finally { endTransaction(); } Setnodes = new HashSet(); nodes.add(A1NodeRef); nodes.add(A2NodeRef); nodes.add(A3NodeRef); nodes.add(A4NodeRef); nodes.add(A5NodeRef); /** * transfer (sync) * check the transfered aspect * create node B6. Fake its transfered aspect to be from Repo B, Non Alien. * transfer (sync) */ startNewTransaction(); try { { TransferDefinition definition = new TransferDefinition(); definition.setNodes(nodes); definition.setSync(true); transferService.transfer(targetName, definition); } } finally { endTransaction(); } startNewTransaction(); try { // Node B6 - faked transfer from repository B. Child of Destination node A1 NodeRef a1Dest = testNodeFactory.getMappedNodeRef(A1NodeRef); assertTrue("dest node A does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A1NodeRef))); assertEquals("dest node A1 From RepositoryId", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A1NodeRef), TransferModel.PROP_FROM_REPOSITORY_ID), repositoryId); assertEquals("dest node A1 Repository Id", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A1NodeRef), TransferModel.PROP_REPOSITORY_ID), repositoryId); assertEquals("dest node A2 From RepositoryId", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A2NodeRef), TransferModel.PROP_FROM_REPOSITORY_ID), repositoryId); assertEquals("dest node A2 Repository Id", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A2NodeRef), TransferModel.PROP_REPOSITORY_ID), repositoryId); assertEquals("dest node A3 From RepositoryId", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A3NodeRef), TransferModel.PROP_FROM_REPOSITORY_ID), repositoryId); assertEquals("dest node A3 Repository Id", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A3NodeRef), TransferModel.PROP_REPOSITORY_ID), repositoryId); assertEquals("dest node A4 From RepositoryId", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A4NodeRef), TransferModel.PROP_FROM_REPOSITORY_ID), repositoryId); assertEquals("dest node A4 Repository Id", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A4NodeRef), TransferModel.PROP_REPOSITORY_ID), repositoryId); assertEquals("dest node A5 From RepositoryId", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A5NodeRef), TransferModel.PROP_FROM_REPOSITORY_ID), repositoryId); assertEquals("dest node A5 Repository Id", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A5NodeRef), TransferModel.PROP_REPOSITORY_ID), repositoryId); ChildAssociationRef child = nodeService.createNode(a1Dest, ContentModel.ASSOC_CONTAINS, QName.createQName("B6"), ContentModel.TYPE_CONTENT); B6NodeRef = child.getChildRef(); nodeService.setProperty(B6NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(B6NodeRef, ContentModel.PROP_NAME, "B6"); /** * The first tranfer was mocked to repository A - this is repository B. */ // This is repository B so there's no need to fake it // nodeService.setProperty(B6NodeRef, TransferModel.PROP_FROM_REPOSITORY_ID, REPO_ID_B); // nodeService.setProperty(B6NodeRef, TransferModel.PROP_REPOSITORY_ID, REPO_ID_B); ContentWriter writer = contentService.getWriter(B6NodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } finally { endTransaction(); } startNewTransaction(); try { TransferDefinition definition = new TransferDefinition(); definition.setNodes(nodes); definition.setSync(true); transferService.transfer(targetName, definition); } finally { endTransaction(); } startNewTransaction(); try { // Does node B6 still exist ? assertTrue("dest node B6 does not exist", nodeService.exists(B6NodeRef)); assertTrue("B6 not alien", nodeService.hasAspect(B6NodeRef, TransferModel.ASPECT_ALIEN)); } finally { endTransaction(); } /** Step 2 * Chain Sync * Change Nodes A1 ... A5 source to be received "from repo B" * Create Node A7 - Fake it to be received "from repo B" * transfer */ String NEW_TITLE="Chain sync"; startNewTransaction(); try { nodeService.setProperty(A1NodeRef, ContentModel.PROP_TITLE, NEW_TITLE); nodeService.setProperty(A1NodeRef, TransferModel.PROP_FROM_REPOSITORY_ID, REPO_ID_B); nodeService.setProperty(A1NodeRef, TransferModel.PROP_REPOSITORY_ID, REPO_ID_B); nodeService.setProperty(A2NodeRef, ContentModel.PROP_TITLE, NEW_TITLE); nodeService.setProperty(A2NodeRef, TransferModel.PROP_FROM_REPOSITORY_ID, REPO_ID_B); nodeService.setProperty(A2NodeRef, TransferModel.PROP_REPOSITORY_ID, REPO_ID_B); nodeService.setProperty(A3NodeRef, ContentModel.PROP_TITLE, NEW_TITLE); nodeService.setProperty(A3NodeRef, TransferModel.PROP_FROM_REPOSITORY_ID, REPO_ID_B); nodeService.setProperty(A3NodeRef, TransferModel.PROP_REPOSITORY_ID, REPO_ID_B); /** * The repository was mocked to repoistory A. This is repository B */ ChildAssociationRef child = nodeService.createNode(A2NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A7"), ContentModel.TYPE_CONTENT); A7NodeRef = child.getChildRef(); nodeService.setProperty(A7NodeRef, ContentModel.PROP_TITLE, NEW_TITLE); nodeService.setProperty(A7NodeRef, ContentModel.PROP_NAME, "A7"); nodeService.setProperty(A7NodeRef, ContentModel.PROP_TITLE, NEW_TITLE); nodeService.setProperty(A7NodeRef, TransferModel.PROP_FROM_REPOSITORY_ID, REPO_ID_B); nodeService.setProperty(A7NodeRef, TransferModel.PROP_REPOSITORY_ID, REPO_ID_B); ContentWriter writer = contentService.getWriter(A3NodeRef, ContentModel.PROP_CONTENT, true); writer.setLocale(CONTENT_LOCALE); writer.putContent(CONTENT_STRING); } finally { endTransaction(); } nodes.add(A7NodeRef); startNewTransaction(); try { TransferDefinition definition = new TransferDefinition(); definition.setNodes(nodes); definition.setSync(true); transferService.transfer(targetName, definition); } finally { endTransaction(); } try { assertTrue("dest node A7 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A7NodeRef))); assertEquals("dest node A1 Title", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A1NodeRef), ContentModel.PROP_TITLE), NEW_TITLE); assertEquals("dest node A1 Repository Id", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A1NodeRef), TransferModel.PROP_REPOSITORY_ID), REPO_ID_B); assertEquals("dest node A1 Repository Id", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A1NodeRef), TransferModel.PROP_FROM_REPOSITORY_ID), repositoryId); assertEquals("dest node A2 Title", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A2NodeRef), ContentModel.PROP_TITLE), NEW_TITLE); assertEquals("dest node A2 Repository Id", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A2NodeRef), TransferModel.PROP_REPOSITORY_ID), REPO_ID_B); assertEquals("dest node A2 Repository Id", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A2NodeRef), TransferModel.PROP_FROM_REPOSITORY_ID), repositoryId); assertEquals("dest node A3 Title", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A3NodeRef), ContentModel.PROP_TITLE), NEW_TITLE); assertEquals("dest node A3 Repository Id", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A3NodeRef), TransferModel.PROP_REPOSITORY_ID), REPO_ID_B); assertEquals("dest node A3 Repository Id", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A3NodeRef), TransferModel.PROP_FROM_REPOSITORY_ID), repositoryId); } finally { endTransaction(); } } // test two repo sync /** * Transfer sync from multiple repos. * * This is a unit test so does lots of shenanigans to fake transfer from three repositories on a single repo. * * Trees of nodes *

     *      A1              B1               C1
     *      |                                 |    
     *    A2/images                       A2 Dummy/images
     *      |                              |
     *      A3                            C3 
     *                                          
     * Destination   
     *      B1
     *      |
     *    A2/images
     *      |     |
     *      C3    A3 
     *            | 
     *            C4         
     * 
* Step 1. Transfer from A to B. * Step 2. Transfer from C to B (crossing over on A2Dest) * Step 3. Invade A3Dest via C * Step 4. Delete C4. Sync from C * Step 5. Delete C3 - A2 dest images folder uninvaded. */ public void testMultiRepoTransfer() throws Exception { setDefaultRollback(false); String CONTENT_TITLE = "ContentTitle"; String CONTENT_TITLE_UPDATED = "ContentTitleUpdated"; Locale CONTENT_LOCALE = Locale.GERMAN; String CONTENT_STRING = "Hello"; String targetName = "testMultiRepoTransfer"; TransferTarget transferMe; NodeRef S0NodeRef; NodeRef A1NodeRef; NodeRef A2NodeRef; NodeRef A3NodeRef; NodeRef B1NodeRef; NodeRef C1NodeRef; NodeRef C2NodeRef; NodeRef C3NodeRef; NodeRef C4NodeRef; NodeRef A3Dummy; startNewTransaction(); try { /** * Get guest home */ String guestHomeQuery = "/app:company_home/app:guest_home"; ResultSet guestHomeResult = searchService.query(StoreRef.STORE_REF_WORKSPACE_SPACESSTORE, SearchService.LANGUAGE_XPATH, guestHomeQuery); assertEquals("", 1, guestHomeResult.length()); NodeRef guestHome = guestHomeResult.getNodeRef(0); { /** * Node Source - located under guest home */ String name = GUID.generate(); ChildAssociationRef child = nodeService.createNode(guestHome, ContentModel.ASSOC_CONTAINS, QName.createQName(name), ContentModel.TYPE_FOLDER); S0NodeRef = child.getChildRef(); nodeService.setProperty(S0NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(S0NodeRef, ContentModel.PROP_NAME, name); } { // Node A1 ChildAssociationRef child = nodeService.createNode(S0NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A1"), ContentModel.TYPE_FOLDER); A1NodeRef = child.getChildRef(); nodeService.setProperty(A1NodeRef, ContentModel.PROP_TITLE, "A1"); nodeService.setProperty(A1NodeRef, ContentModel.PROP_NAME, "A1"); } { // Node A2 ChildAssociationRef child = nodeService.createNode(A1NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("images"), ContentModel.TYPE_FOLDER); A2NodeRef = child.getChildRef(); nodeService.setProperty(A2NodeRef, ContentModel.PROP_TITLE, "images"); nodeService.setProperty(A2NodeRef, ContentModel.PROP_NAME, "images"); } { // Node A3 ChildAssociationRef child = nodeService.createNode(A2NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A3"), ContentModel.TYPE_FOLDER); A3NodeRef = child.getChildRef(); nodeService.setProperty(A3NodeRef, ContentModel.PROP_TITLE, "A3"); nodeService.setProperty(A3NodeRef, ContentModel.PROP_NAME, "A3"); } { // Node B1 ChildAssociationRef child = nodeService.createNode(S0NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("B1"), ContentModel.TYPE_FOLDER); B1NodeRef = child.getChildRef(); nodeService.setProperty(B1NodeRef, ContentModel.PROP_TITLE, "B1"); nodeService.setProperty(B1NodeRef, ContentModel.PROP_NAME, "B1"); } { // Node C1 ChildAssociationRef child = nodeService.createNode(S0NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("C1"), ContentModel.TYPE_FOLDER); C1NodeRef = child.getChildRef(); nodeService.setProperty(C1NodeRef, ContentModel.PROP_TITLE, "C1"); nodeService.setProperty(C1NodeRef, ContentModel.PROP_NAME, "C1"); } { // Node C2/images ChildAssociationRef child = nodeService.createNode(C1NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("images"), ContentModel.TYPE_FOLDER); C2NodeRef = child.getChildRef(); nodeService.setProperty(C2NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(C2NodeRef, ContentModel.PROP_NAME, "images"); } { // Node C3 ChildAssociationRef child = nodeService.createNode(C2NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("C3"), ContentModel.TYPE_FOLDER); C3NodeRef = child.getChildRef(); nodeService.setProperty(C3NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(C3NodeRef, ContentModel.PROP_NAME, "C3"); } { // Node A3 (Dummy) ChildAssociationRef child = nodeService.createNode(C2NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A3"), ContentModel.TYPE_FOLDER); A3Dummy = child.getChildRef(); nodeService.setProperty(A3Dummy, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(A3Dummy, ContentModel.PROP_NAME, "A3 Dummy"); } { // Node C4 ChildAssociationRef child = nodeService.createNode(A3Dummy, ContentModel.ASSOC_CONTAINS, QName.createQName("C4"), ContentModel.TYPE_FOLDER); C4NodeRef = child.getChildRef(); nodeService.setProperty(C4NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(C4NodeRef, ContentModel.PROP_NAME, "C4"); } // Create the transfer target if it does not already exist if(!transferService.targetExists(targetName)) { transferMe = createTransferTarget(targetName); } else { transferMe = transferService.getTransferTarget(targetName); } } finally { endTransaction(); } /** * For unit test * - replace the HTTP transport with the in-process transport * - Map path from A1 to B1 (So transfer will transfer by path) * - Map path from C1 to B1 */ TransferTransmitter transmitter = new UnitTestInProcessTransmitterImpl(receiver, contentService, transactionService); transferServiceImpl.setTransmitter(transmitter); UnitTestTransferManifestNodeFactory testNodeFactory = new UnitTestTransferManifestNodeFactory(this.transferManifestNodeFactory); transferServiceImpl.setTransferManifestNodeFactory(testNodeFactory); List> pathMap = testNodeFactory.getPathMap(); // Map Project A/images to Project B/images // Map Project C/images to Project A/images nodeService.getPath(A2NodeRef); pathMap.add(new Pair(nodeService.getPath(A1NodeRef), nodeService.getPath(B1NodeRef))); pathMap.add(new Pair(nodeService.getPath(C1NodeRef), nodeService.getPath(B1NodeRef))); DescriptorService mockedDescriptorService = getMockDescriptorService(REPO_ID_A); transferServiceImpl.setDescriptorService(mockedDescriptorService); /** * Step 1 * Now transfer in A's nodes to Repo B */ startNewTransaction(); try { TransferDefinition definition = new TransferDefinition(); Collection nodes = new ArrayList(); nodes.add(A1NodeRef); nodes.add(A2NodeRef); nodes.add(A3NodeRef); definition.setNodes(nodes); definition.setSync(true); transferService.transfer(targetName, definition); } finally { endTransaction(); } startNewTransaction(); try { assertTrue("dest node A2 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A2NodeRef))); assertTrue("dest node A3 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A3NodeRef))); // Check that A3 dest is a child of A2Dest which is a child of B1 ChildAssociationRef A3Ref = nodeService.getPrimaryParent(testNodeFactory.getMappedNodeRef(A3NodeRef)); assertEquals("A3 dest is connected to the wrong node", A3Ref.getParentRef(), testNodeFactory.getMappedNodeRef(A2NodeRef)); ChildAssociationRef A2Ref = nodeService.getPrimaryParent(testNodeFactory.getMappedNodeRef(A2NodeRef)); assertEquals("A2 dest is connected to the wrong node", A2Ref.getParentRef(), B1NodeRef); assertEquals("A2 dest owned by wrong repo", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A2NodeRef), TransferModel.PROP_FROM_REPOSITORY_ID), REPO_ID_A); assertEquals("A3 dest owned by wrong repo", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A3NodeRef), TransferModel.PROP_FROM_REPOSITORY_ID), REPO_ID_A); } finally { endTransaction(); } /** * Step 2 * Now transfer in C's nodes * B2 (Owned by A) gets invaded by C */ startNewTransaction(); try { mockedDescriptorService = getMockDescriptorService(REPO_ID_C); transferServiceImpl.setDescriptorService(mockedDescriptorService); TransferDefinition definition = new TransferDefinition(); Collection nodes = new ArrayList(); nodes.add(C1NodeRef); nodes.add(C2NodeRef); nodes.add(C3NodeRef); definition.setNodes(nodes); definition.setSync(true); transferService.transfer(targetName, definition); } finally { endTransaction(); } startNewTransaction(); try { assertTrue("dest node A3 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A3NodeRef))); assertTrue("dest node C3 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(C3NodeRef))); // Check that A3 dest is a child of A2Dest which is a child of B1 // Check that C3 dest is a child of A2Dest ChildAssociationRef A3Ref = nodeService.getPrimaryParent(testNodeFactory.getMappedNodeRef(A3NodeRef)); assertEquals("A3 dest is connected to the wrong node", A3Ref.getParentRef(), testNodeFactory.getMappedNodeRef(A2NodeRef)); ChildAssociationRef C3Ref = nodeService.getPrimaryParent(testNodeFactory.getMappedNodeRef(A3NodeRef)); assertEquals("C3 dest is connected to the wrong node", C3Ref.getParentRef(), testNodeFactory.getMappedNodeRef(A2NodeRef)); ChildAssociationRef A2Ref = nodeService.getPrimaryParent(testNodeFactory.getMappedNodeRef(A2NodeRef)); assertEquals("A2 dest is connected to the wrong node", A2Ref.getParentRef(), B1NodeRef); assertTrue("A2 dest is not invaded", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(A2NodeRef), TransferModel.ASPECT_ALIEN)); assertTrue("C3 dest is not invaded", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(C3NodeRef), TransferModel.ASPECT_ALIEN)); assertFalse("A3 dest is invaded", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(A3NodeRef), TransferModel.ASPECT_ALIEN)); assertEquals("A2 dest owned by wrong repo", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A2NodeRef), TransferModel.PROP_FROM_REPOSITORY_ID), REPO_ID_A); assertEquals("A3 dest owned by wrong repo", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A3NodeRef), TransferModel.PROP_FROM_REPOSITORY_ID), REPO_ID_A); assertEquals("C3 dest owned by wrong repo", nodeService.getProperty(testNodeFactory.getMappedNodeRef(C3NodeRef), TransferModel.PROP_FROM_REPOSITORY_ID), REPO_ID_C); } finally { endTransaction(); } /** * Step 3 * Invade A3Dest via transfer of C4 from C */ startNewTransaction(); try { mockedDescriptorService = getMockDescriptorService(REPO_ID_C); transferServiceImpl.setDescriptorService(mockedDescriptorService); TransferDefinition definition = new TransferDefinition(); Collection nodes = new ArrayList(); nodes.add(C4NodeRef); definition.setNodes(nodes); definition.setSync(false); transferService.transfer(targetName, definition); } finally { endTransaction(); } startNewTransaction(); try { assertTrue("dest node A3 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A3NodeRef))); assertTrue("dest node C3 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(C3NodeRef))); assertTrue("dest node C4 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(C4NodeRef))); assertTrue("C4 is not an invader", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(C4NodeRef), TransferModel.ASPECT_ALIEN)); assertTrue("A3 is not an invader", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(A3NodeRef), TransferModel.ASPECT_ALIEN)); assertEquals("A2 dest owned by wrong repo", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A2NodeRef), TransferModel.PROP_FROM_REPOSITORY_ID), REPO_ID_A); assertEquals("A3 dest owned by wrong repo", nodeService.getProperty(testNodeFactory.getMappedNodeRef(A3NodeRef), TransferModel.PROP_FROM_REPOSITORY_ID), REPO_ID_A); assertEquals("C3 dest owned by wrong repo", nodeService.getProperty(testNodeFactory.getMappedNodeRef(C3NodeRef), TransferModel.PROP_FROM_REPOSITORY_ID), REPO_ID_C); } finally { endTransaction(); } /** * Step 4 * Uninvade A3 from C by deleting C4 * Via Sync of A3Dummy (which has the same destination path as A3). */ startNewTransaction(); try { nodeService.deleteNode(C4NodeRef); } finally { endTransaction(); } startNewTransaction(); try { mockedDescriptorService = getMockDescriptorService(REPO_ID_C); transferServiceImpl.setDescriptorService(mockedDescriptorService); TransferDefinition definition = new TransferDefinition(); Collection nodes = new ArrayList(); nodes.add(A3Dummy); definition.setNodes(nodes); definition.setSync(true); transferService.transfer(targetName, definition); } finally { endTransaction(); } startNewTransaction(); try { assertTrue("dest node A3 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A3NodeRef))); assertTrue("dest node C3 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(C3NodeRef))); assertFalse("dest node C4 not deleted", nodeService.exists(testNodeFactory.getMappedNodeRef(C4NodeRef))); logger.debug("A3 Dest is " + testNodeFactory.getMappedNodeRef(A3NodeRef)); assertFalse("A3 Dest still invaded by C4", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(A3NodeRef), TransferModel.ASPECT_ALIEN)); } finally { endTransaction(); } /** * Step 5 - repeat the above test with transfer(non sync) rather than transfer(sync) * Uninvade by deleting C3. */ startNewTransaction(); try { nodeService.deleteNode(C3NodeRef); } finally { endTransaction(); } startNewTransaction(); try { mockedDescriptorService = getMockDescriptorService(REPO_ID_C); transferServiceImpl.setDescriptorService(mockedDescriptorService); TransferDefinition definition = new TransferDefinition(); Collection nodes = new ArrayList(); NodeRef C3Deleted = new NodeRef(StoreRef.STORE_REF_ARCHIVE_SPACESSTORE, C3NodeRef.getId()); nodes.add(C3Deleted); definition.setNodes(nodes); definition.setSync(false); transferService.transfer(targetName, definition); } finally { endTransaction(); } startNewTransaction(); try { assertTrue("dest node A3 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A3NodeRef))); assertFalse("dest node C3 not deleted", nodeService.exists(testNodeFactory.getMappedNodeRef(C3NodeRef))); assertFalse("dest node C4 not deleted", nodeService.exists(testNodeFactory.getMappedNodeRef(C4NodeRef))); assertFalse("A3 still invaded", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(A3NodeRef), TransferModel.ASPECT_ALIEN)); assertFalse("A2 still invaded", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(A2NodeRef), TransferModel.ASPECT_ALIEN)); } finally { endTransaction(); } } // test multi repo sync // Utility methods below. private TransferTarget createTransferTarget(String name) { String title = "title"; String description = "description"; String endpointProtocol = "http"; String endpointHost = "MARKR02"; int endpointPort = 7080; String endpointPath = "/alfresco/service/api/transfer"; String username = "admin"; char[] password = "admin".toCharArray(); /** * Now go ahead and create our first transfer target */ TransferTarget target = transferService.createAndSaveTransferTarget(name, title, description, endpointProtocol, endpointHost, endpointPort, endpointPath, username, password); return target; } /** * Test the transfer method behaviour with respect to move and alien nodes. * * So we have Repository A transferring content and Repository B is the local repo that we * move alien nodes in and out. * * Tree *
     *         B1
     *    |          |
     *    C2(p1)    C3(p2)
     *    |
     *    A4
     * 
* * Setup tree above. Validate that A1 is child of C2. * * Step 1. Move A4 fron C2 to C3 via transfer. * C2Dest should stop being invaded C3Dest should be invaded. */ public void testMultiRepoTransferMove() throws Exception { setDefaultRollback(false); String CONTENT_TITLE = "ContentTitle"; String CONTENT_TITLE_UPDATED = "ContentTitleUpdated"; Locale CONTENT_LOCALE = Locale.GERMAN; String CONTENT_STRING = "Hello"; String targetName = "testMultiRepoTransferMove"; TransferTarget transferMe; NodeRef S0NodeRef; NodeRef A1NodeRef; NodeRef B1NodeRef; NodeRef C1NodeRef; NodeRef C2NodeRef; NodeRef C3NodeRef; NodeRef A4NodeRef; NodeRef C2DummyNodeRef; NodeRef C3DummyNodeRef; QName C2Path = QName.createQName("p2"); QName C3Path= QName.createQName("p3"); startNewTransaction(); try { /** * Get guest home */ String guestHomeQuery = "/app:company_home/app:guest_home"; ResultSet guestHomeResult = searchService.query(StoreRef.STORE_REF_WORKSPACE_SPACESSTORE, SearchService.LANGUAGE_XPATH, guestHomeQuery); assertEquals("", 1, guestHomeResult.length()); NodeRef guestHome = guestHomeResult.getNodeRef(0); { /** * Node Source - located under guest home */ String name = GUID.generate(); ChildAssociationRef child = nodeService.createNode(guestHome, ContentModel.ASSOC_CONTAINS, QName.createQName(name), ContentModel.TYPE_FOLDER); S0NodeRef = child.getChildRef(); nodeService.setProperty(S0NodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(S0NodeRef, ContentModel.PROP_NAME, name); } { // Node A1 ChildAssociationRef child = nodeService.createNode(S0NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("A1"), ContentModel.TYPE_FOLDER); A1NodeRef = child.getChildRef(); nodeService.setProperty(A1NodeRef, ContentModel.PROP_TITLE, "A1"); nodeService.setProperty(A1NodeRef, ContentModel.PROP_NAME, "A1"); } { // Node B1 ChildAssociationRef child = nodeService.createNode(S0NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("B1"), ContentModel.TYPE_FOLDER); B1NodeRef = child.getChildRef(); nodeService.setProperty(B1NodeRef, ContentModel.PROP_TITLE, "B1"); nodeService.setProperty(B1NodeRef, ContentModel.PROP_NAME, "B1"); } { // Node C1 ChildAssociationRef child = nodeService.createNode(S0NodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("C1"), ContentModel.TYPE_FOLDER); C1NodeRef = child.getChildRef(); nodeService.setProperty(C1NodeRef, ContentModel.PROP_TITLE, "C1"); nodeService.setProperty(C1NodeRef, ContentModel.PROP_NAME, "C1"); } { // Node C2 ChildAssociationRef child = nodeService.createNode(C1NodeRef, ContentModel.ASSOC_CONTAINS, C2Path, ContentModel.TYPE_FOLDER); C2NodeRef = child.getChildRef(); nodeService.setProperty(C2NodeRef, ContentModel.PROP_TITLE, "C2"); nodeService.setProperty(C2NodeRef, ContentModel.PROP_NAME, "C2"); } { // Node C3 ChildAssociationRef child = nodeService.createNode(C1NodeRef, ContentModel.ASSOC_CONTAINS, C3Path, ContentModel.TYPE_FOLDER); C3NodeRef = child.getChildRef(); nodeService.setProperty(C3NodeRef, ContentModel.PROP_TITLE, "C3"); nodeService.setProperty(C3NodeRef, ContentModel.PROP_NAME, "C3"); } { // Node C2 (Dummy) ChildAssociationRef child = nodeService.createNode(A1NodeRef, ContentModel.ASSOC_CONTAINS, C2Path, ContentModel.TYPE_FOLDER); C2DummyNodeRef = child.getChildRef(); nodeService.setProperty(C2DummyNodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(C2DummyNodeRef, ContentModel.PROP_NAME, "C2 Dummy"); } { // Node C3 (Dummy) ChildAssociationRef child = nodeService.createNode(A1NodeRef, ContentModel.ASSOC_CONTAINS, C3Path, ContentModel.TYPE_FOLDER); C3DummyNodeRef = child.getChildRef(); nodeService.setProperty(C3DummyNodeRef, ContentModel.PROP_TITLE, CONTENT_TITLE); nodeService.setProperty(C3DummyNodeRef, ContentModel.PROP_NAME, "C3 Dummy"); } { // Node A4 ChildAssociationRef child = nodeService.createNode(C2DummyNodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("C4"), ContentModel.TYPE_FOLDER); A4NodeRef = child.getChildRef(); nodeService.setProperty(A4NodeRef, ContentModel.PROP_TITLE, "C4"); nodeService.setProperty(A4NodeRef, ContentModel.PROP_NAME, "C4"); } // Create the transfer target if it does not already exist if(!transferService.targetExists(targetName)) { transferMe = createTransferTarget(targetName); } else { transferMe = transferService.getTransferTarget(targetName); } } finally { endTransaction(); } /** * For unit test * - replace the HTTP transport with the in-process transport * - Map path from A1 to B1 (So transfer will transfer by path) * - Map path from C1 to B1 */ TransferTransmitter transmitter = new UnitTestInProcessTransmitterImpl(receiver, contentService, transactionService); transferServiceImpl.setTransmitter(transmitter); UnitTestTransferManifestNodeFactory testNodeFactory = new UnitTestTransferManifestNodeFactory(this.transferManifestNodeFactory); transferServiceImpl.setTransferManifestNodeFactory(testNodeFactory); List> pathMap = testNodeFactory.getPathMap(); // Map Project A to Project B // Map Project C to Project B pathMap.add(new Pair(nodeService.getPath(A1NodeRef), nodeService.getPath(B1NodeRef))); pathMap.add(new Pair(nodeService.getPath(C1NodeRef), nodeService.getPath(B1NodeRef))); DescriptorService mockedDescriptorService = getMockDescriptorService(REPO_ID_C); transferServiceImpl.setDescriptorService(mockedDescriptorService); /** * Step 1 * Now transfer in C's nodes to Repo B */ startNewTransaction(); try { TransferDefinition definition = new TransferDefinition(); Collection nodes = new ArrayList(); nodes.add(C1NodeRef); nodes.add(C2NodeRef); nodes.add(C3NodeRef); definition.setNodes(nodes); definition.setSync(true); transferService.transfer(targetName, definition); } finally { endTransaction(); } startNewTransaction(); try { assertTrue("dest node C2 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(C2NodeRef))); assertTrue("dest node C3 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(C3NodeRef))); // Check that C3 dest is a child of B1 ChildAssociationRef C3Ref = nodeService.getPrimaryParent(testNodeFactory.getMappedNodeRef(C3NodeRef)); assertEquals("A3 dest is connected to the wrong node", C3Ref.getParentRef(), B1NodeRef); ChildAssociationRef C2Ref = nodeService.getPrimaryParent(testNodeFactory.getMappedNodeRef(C2NodeRef)); assertEquals("A2 dest is connected to the wrong node", C2Ref.getParentRef(), B1NodeRef); } finally { endTransaction(); } mockedDescriptorService = getMockDescriptorService(REPO_ID_A); transferServiceImpl.setDescriptorService(mockedDescriptorService); /** * Step 2 * Now transfer in A's nodes * C2 (Dest) gets invaded by A4 */ startNewTransaction(); try { TransferDefinition definition = new TransferDefinition(); Collection nodes = new ArrayList(); nodes.add(A4NodeRef); definition.setNodes(nodes); definition.setSync(true); transferService.transfer(targetName, definition); } finally { endTransaction(); } startNewTransaction(); try { assertTrue("dest node A4 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A4NodeRef))); assertTrue("dest node C3 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(C3NodeRef))); assertTrue("dest node C2 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(C2NodeRef))); // Check that A4 dest is a child of C2Dest which is a child of B1 ChildAssociationRef A4Ref = nodeService.getPrimaryParent(testNodeFactory.getMappedNodeRef(A4NodeRef)); assertEquals("A4 dest is connected to the wrong node", A4Ref.getParentRef(), testNodeFactory.getMappedNodeRef(C2NodeRef)); assertTrue("C2 dest is not invaded", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(C2NodeRef), TransferModel.ASPECT_ALIEN)); assertFalse("C3 dest is not invaded", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(C3NodeRef), TransferModel.ASPECT_ALIEN)); } finally { endTransaction(); } /** * Step 3 * Now move A3 * C2 (Dest) gets invaded by A4 */ startNewTransaction(); try { nodeService.moveNode(A4NodeRef, C3DummyNodeRef, ContentModel.ASSOC_CONTAINS, QName.createQName("C4")); } finally { endTransaction(); } startNewTransaction(); try { TransferDefinition definition = new TransferDefinition(); Collection nodes = new ArrayList(); nodes.add(A4NodeRef); definition.setNodes(nodes); definition.setSync(true); transferService.transfer(targetName, definition); } finally { endTransaction(); } startNewTransaction(); try { assertTrue("dest node A4 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(A4NodeRef))); assertTrue("dest node C3 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(C3NodeRef))); assertTrue("dest node C2 does not exist", nodeService.exists(testNodeFactory.getMappedNodeRef(C2NodeRef))); // Check that A4 dest is a child of C3Dest which is a child of B1 ChildAssociationRef A4Ref = nodeService.getPrimaryParent(testNodeFactory.getMappedNodeRef(A4NodeRef)); assertEquals("A4 dest is connected to the wrong node", A4Ref.getParentRef(), testNodeFactory.getMappedNodeRef(C3NodeRef)); assertTrue("A4 dest is not invaded", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(A4NodeRef), TransferModel.ASPECT_ALIEN)); assertTrue("C3 dest is not invaded", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(C3NodeRef), TransferModel.ASPECT_ALIEN)); assertFalse("C2 dest is still invaded", nodeService.hasAspect(testNodeFactory.getMappedNodeRef(C2NodeRef), TransferModel.ASPECT_ALIEN)); } finally { endTransaction(); } } /** * transfer should only be able to update and delete nodes that are "from" the transferring system */ public void testFromRepo() { assertTrue("not yet implemented", false); } private void createUser(String userName, String password) { if (this.authenticationService.authenticationExists(userName) == false) { this.authenticationService.createAuthentication(userName, password.toCharArray()); PropertyMap ppOne = new PropertyMap(4); ppOne.put(ContentModel.PROP_USERNAME, userName); ppOne.put(ContentModel.PROP_FIRSTNAME, "firstName"); ppOne.put(ContentModel.PROP_LASTNAME, "lastName"); ppOne.put(ContentModel.PROP_EMAIL, "email@email.com"); ppOne.put(ContentModel.PROP_JOBTITLE, "jobTitle"); this.personService.createPerson(ppOne); } } private DescriptorService getMockDescriptorService(String repositoryId) { DescriptorService descriptorService = mock(DescriptorService.class); Descriptor descriptor = mock(Descriptor.class); when(descriptor.getId()).thenReturn(repositoryId); when(descriptorService.getCurrentRepositoryDescriptor()).thenReturn(descriptor); return descriptorService; } }