mirror of
https://github.com/Alfresco/alfresco-community-repo.git
synced 2025-06-30 18:15:39 +00:00
76890: Merged PLATFORM1 (5.0/Cloud) to HEAD-BUG-FIX (5.0/Cloud) 65482: Merge DEV to PLATFORM1 ACE-502 - Transfer service : implement transferring of categories and tags 58825 : MNT-8098 - Create categories on target runs as "System". Bug fix when removing categories which results in a "null" node ref. 58716 : MNT-8098 - transfer of categories and tags. git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@77731 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
1447 lines
57 KiB
Java
1447 lines
57 KiB
Java
/*
|
|
* Copyright (C) 2009-2011 Alfresco Software Limited.
|
|
*
|
|
* This file is part of Alfresco
|
|
*
|
|
* Alfresco is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Lesser General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* Alfresco is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
* along with Alfresco. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
package org.alfresco.repo.transfer;
|
|
|
|
import java.io.BufferedReader;
|
|
import java.io.BufferedWriter;
|
|
import java.io.File;
|
|
import java.io.FileInputStream;
|
|
import java.io.FileOutputStream;
|
|
import java.io.IOException;
|
|
import java.io.InputStreamReader;
|
|
import java.io.OutputStreamWriter;
|
|
import java.io.Serializable;
|
|
import java.io.Writer;
|
|
import java.text.SimpleDateFormat;
|
|
import java.util.Arrays;
|
|
import java.util.Collection;
|
|
import java.util.Collections;
|
|
import java.util.Date;
|
|
import java.util.HashMap;
|
|
import java.util.HashSet;
|
|
import java.util.LinkedList;
|
|
import java.util.List;
|
|
import java.util.Map;
|
|
import java.util.Set;
|
|
import java.util.TreeMap;
|
|
|
|
import javax.transaction.UserTransaction;
|
|
import javax.xml.parsers.ParserConfigurationException;
|
|
import javax.xml.parsers.SAXParser;
|
|
import javax.xml.parsers.SAXParserFactory;
|
|
|
|
import org.alfresco.error.AlfrescoRuntimeException;
|
|
import org.alfresco.model.ContentModel;
|
|
import org.alfresco.repo.cache.SimpleCache;
|
|
import org.alfresco.repo.transaction.RetryingTransactionHelper.RetryingTransactionCallback;
|
|
import org.alfresco.repo.transfer.manifest.TransferManifestDeletedNode;
|
|
import org.alfresco.repo.transfer.manifest.TransferManifestHeader;
|
|
import org.alfresco.repo.transfer.manifest.TransferManifestNode;
|
|
import org.alfresco.repo.transfer.manifest.TransferManifestNodeFactory;
|
|
import org.alfresco.repo.transfer.manifest.TransferManifestNodeHelper;
|
|
import org.alfresco.repo.transfer.manifest.TransferManifestNormalNode;
|
|
import org.alfresco.repo.transfer.manifest.TransferManifestProcessor;
|
|
import org.alfresco.repo.transfer.manifest.TransferManifestWriter;
|
|
import org.alfresco.repo.transfer.manifest.XMLTransferManifestReader;
|
|
import org.alfresco.repo.transfer.manifest.XMLTransferManifestWriter;
|
|
import org.alfresco.repo.transfer.report.TransferReporter;
|
|
import org.alfresco.repo.transfer.requisite.DeltaListRequsiteProcessor;
|
|
import org.alfresco.repo.transfer.requisite.XMLTransferRequsiteReader;
|
|
import org.alfresco.service.cmr.action.Action;
|
|
import org.alfresco.service.cmr.action.ActionService;
|
|
import org.alfresco.service.cmr.repository.ChildAssociationRef;
|
|
import org.alfresco.service.cmr.repository.ContentData;
|
|
import org.alfresco.service.cmr.repository.NodeRef;
|
|
import org.alfresco.service.cmr.repository.NodeService;
|
|
import org.alfresco.service.cmr.repository.StoreRef;
|
|
import org.alfresco.service.cmr.search.SearchService;
|
|
import org.alfresco.service.cmr.transfer.TransferCallback;
|
|
import org.alfresco.service.cmr.transfer.TransferCancelledException;
|
|
import org.alfresco.service.cmr.transfer.TransferDefinition;
|
|
import org.alfresco.service.cmr.transfer.TransferEndEvent;
|
|
import org.alfresco.service.cmr.transfer.TransferEvent;
|
|
import org.alfresco.service.cmr.transfer.TransferEventCancelled;
|
|
import org.alfresco.service.cmr.transfer.TransferEventError;
|
|
import org.alfresco.service.cmr.transfer.TransferEventReport;
|
|
import org.alfresco.service.cmr.transfer.TransferEventSuccess;
|
|
import org.alfresco.service.cmr.transfer.TransferException;
|
|
import org.alfresco.service.cmr.transfer.TransferFailureException;
|
|
import org.alfresco.service.cmr.transfer.TransferProgress;
|
|
import org.alfresco.service.cmr.transfer.TransferService2;
|
|
import org.alfresco.service.cmr.transfer.TransferTarget;
|
|
import org.alfresco.service.cmr.transfer.TransferVersion;
|
|
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.namespace.RegexQNamePattern;
|
|
import org.alfresco.service.transaction.TransactionService;
|
|
import org.alfresco.util.PropertyCheck;
|
|
import org.alfresco.util.TempFileProvider;
|
|
import org.apache.commons.logging.Log;
|
|
import org.apache.commons.logging.LogFactory;
|
|
import org.xml.sax.SAXException;
|
|
|
|
|
|
/**
|
|
* Implementation of the Transfer Service.
|
|
*
|
|
* @author Mark Rogers
|
|
*
|
|
*/
|
|
public class TransferServiceImpl2 implements TransferService2
|
|
{
|
|
private static Log logger = LogFactory.getLog(TransferServiceImpl2.class);
|
|
|
|
private static final String MSG_NO_HOME = "transfer_service.unable_to_find_transfer_home";
|
|
private static final String MSG_NO_GROUP = "transfer_service.unable_to_find_transfer_group";
|
|
private static final String MSG_NO_TARGET = "transfer_service.unable_to_find_transfer_target";
|
|
private static final String MSG_ERR_TRANSFER_ASYNC = "transfer_service.unable_to_transfer_async";
|
|
private static final String MSG_TARGET_EXISTS = "transfer_service.target_exists";
|
|
private static final String MSG_NO_NODES = "transfer_service.no_nodes";
|
|
private static final String MSG_MISSING_ENDPOINT_PATH = "transfer_service.missing_endpoint_path";
|
|
private static final String MSG_MISSING_ENDPOINT_PROTOCOL = "transfer_service.missing_endpoint_protocol";
|
|
private static final String MSG_MISSING_ENDPOINT_HOST = "transfer_service.missing_endpoint_host";
|
|
private static final String MSG_MISSING_ENDPOINT_PORT = "transfer_service.missing_endpoint_port";
|
|
private static final String MSG_MISSING_ENDPOINT_USERNAME = "transfer_service.missing_endpoint_username";
|
|
private static final String MSG_MISSING_ENDPOINT_PASSWORD = "transfer_service.missing_endpoint_password";
|
|
private static final String MSG_FAILED_TO_GET_TRANSFER_STATUS = "transfer_service.failed_to_get_transfer_status";
|
|
private static final String MSG_TARGET_ERROR = "transfer_service.target_error";
|
|
private static final String MSG_UNKNOWN_TARGET_ERROR = "transfer_service.unknown_target_error";
|
|
private static final String MSG_TARGET_NOT_ENABLED = "transfer_service.target_not_enabled";
|
|
private static final String MSG_INCOMPATIBLE_VERSIONS = "transfer_service.incompatible_versions";
|
|
|
|
private static final String FILE_DIRECTORY = "transfer";
|
|
private static final String FILE_SUFFIX = ".xml";
|
|
|
|
private enum ClientTransferState { Begin, Prepare, Commit, Poll, Cancel, Finished, Exit; };
|
|
|
|
/**
|
|
* The synchronised list of transfers in progress.
|
|
*/
|
|
private Map<String, TransferStatus> transferMonitoring = Collections.synchronizedMap(new TreeMap<String,TransferStatus>());
|
|
|
|
public void init()
|
|
{
|
|
PropertyCheck.mandatory(this, "nodeService", nodeService);
|
|
PropertyCheck.mandatory(this, "searchService", searchService);
|
|
PropertyCheck.mandatory(this, "transferSpaceQuery", transferSpaceQuery);
|
|
PropertyCheck.mandatory(this, "defaultTransferGroup", defaultTransferGroup);
|
|
PropertyCheck.mandatory(this, "transmitter", transmitter);
|
|
PropertyCheck.mandatory(this, "namespaceResolver", transmitter);
|
|
PropertyCheck.mandatory(this, "actionService", actionService);
|
|
PropertyCheck.mandatory(this, "transactionService", transactionService);
|
|
PropertyCheck.mandatory(this, "descriptorService", descriptorService);
|
|
PropertyCheck.mandatory(this, "transferVersionChecker", transferVersionChecker);
|
|
}
|
|
|
|
private String transferSpaceQuery;
|
|
private String defaultTransferGroup;
|
|
private NodeService nodeService;
|
|
private SearchService searchService;
|
|
private TransferTransmitter transmitter;
|
|
private TransactionService transactionService;
|
|
private ActionService actionService;
|
|
private TransferManifestNodeFactory transferManifestNodeFactory;
|
|
private TransferReporter transferReporter;
|
|
private DescriptorService descriptorService;
|
|
private TransferVersionChecker transferVersionChecker;
|
|
private NamespaceService namespaceService;
|
|
|
|
/**
|
|
* How long to delay while polling for commit status.
|
|
*/
|
|
private long commitPollDelay = 2000;
|
|
|
|
/**
|
|
* Create a new in memory transfer target
|
|
*/
|
|
public TransferTarget createTransferTarget(String name)
|
|
{
|
|
NodeRef dummy = lookupTransferTarget(name);
|
|
if(dummy != null)
|
|
{
|
|
throw new TransferException(MSG_TARGET_EXISTS, new Object[]{name} );
|
|
}
|
|
|
|
TransferTargetImpl newTarget = new TransferTargetImpl();
|
|
newTarget.setName(name);
|
|
return newTarget;
|
|
}
|
|
|
|
/**
|
|
* create transfer target
|
|
*/
|
|
public TransferTarget createAndSaveTransferTarget(String name, String title, String description, String endpointProtocol, String endpointHost, int endpointPort, String endpointPath, String username, char[] password)
|
|
{
|
|
TransferTargetImpl newTarget = new TransferTargetImpl();
|
|
newTarget.setName(name);
|
|
newTarget.setTitle(title);
|
|
newTarget.setDescription(description);
|
|
newTarget.setEndpointProtocol(endpointProtocol);
|
|
newTarget.setEndpointHost(endpointHost);
|
|
newTarget.setEndpointPort(endpointPort);
|
|
newTarget.setEndpointPath(endpointPath);
|
|
newTarget.setUsername(username);
|
|
newTarget.setPassword(password);
|
|
return createTransferTarget(newTarget);
|
|
|
|
}
|
|
|
|
/**
|
|
* create transfer target
|
|
*/
|
|
private TransferTarget createTransferTarget(TransferTarget newTarget)
|
|
{
|
|
/**
|
|
* Check whether name is already used
|
|
*/
|
|
NodeRef dummy = lookupTransferTarget(newTarget.getName());
|
|
if (dummy != null) { throw new TransferException(MSG_TARGET_EXISTS,
|
|
new Object[] { newTarget.getName() }); }
|
|
|
|
Map<QName, Serializable> properties = new HashMap<QName, Serializable>();
|
|
|
|
// type properties
|
|
properties.put(TransferModel.PROP_ENDPOINT_HOST, newTarget.getEndpointHost());
|
|
properties.put(TransferModel.PROP_ENDPOINT_PORT, newTarget.getEndpointPort());
|
|
properties.put(TransferModel.PROP_ENDPOINT_PROTOCOL, newTarget.getEndpointProtocol());
|
|
properties.put(TransferModel.PROP_ENDPOINT_PATH, newTarget.getEndpointPath());
|
|
properties.put(TransferModel.PROP_USERNAME, newTarget.getUsername());
|
|
properties.put(TransferModel.PROP_PASSWORD, new String(encrypt(newTarget.getPassword())));
|
|
|
|
// titled aspect
|
|
properties.put(ContentModel.PROP_TITLE, newTarget.getTitle());
|
|
properties.put(ContentModel.PROP_NAME, newTarget.getName());
|
|
properties.put(ContentModel.PROP_DESCRIPTION, newTarget.getDescription());
|
|
|
|
// enableable aspect
|
|
properties.put(TransferModel.PROP_ENABLED, Boolean.TRUE);
|
|
|
|
NodeRef defaultGroup = getDefaultGroup();
|
|
|
|
/**
|
|
* Go ahead and create the new node
|
|
*/
|
|
ChildAssociationRef ref = nodeService.createNode(defaultGroup, ContentModel.ASSOC_CONTAINS,
|
|
QName.createQName(TransferModel.TRANSFER_MODEL_1_0_URI, newTarget.getName()),
|
|
TransferModel.TYPE_TRANSFER_TARGET, properties);
|
|
|
|
/**
|
|
* Now create a new TransferTarget object to return to the caller.
|
|
*/
|
|
TransferTargetImpl retVal = new TransferTargetImpl();
|
|
mapTransferTarget(ref.getChildRef(), retVal);
|
|
|
|
return retVal;
|
|
}
|
|
|
|
protected NodeRef getDefaultGroup()
|
|
{
|
|
NodeRef home = getTransferHome();
|
|
List<ChildAssociationRef> refs = nodeService.getChildAssocs(home, ContentModel.ASSOC_CONTAINS, QName.createQName(defaultTransferGroup, namespaceService));
|
|
if (refs.isEmpty())
|
|
{
|
|
// No transfer group.
|
|
throw new TransferException(MSG_NO_GROUP, new Object[]
|
|
{
|
|
defaultTransferGroup
|
|
});
|
|
}
|
|
return refs.get(0).getChildRef();
|
|
}
|
|
|
|
/**
|
|
* Get all transfer targets
|
|
*/
|
|
public Set<TransferTarget> getTransferTargets()
|
|
{
|
|
NodeRef home = getTransferHome();
|
|
|
|
Set<TransferTarget> ret = new HashSet<TransferTarget>();
|
|
|
|
// get all groups
|
|
List<ChildAssociationRef> groups = nodeService.getChildAssocs(home);
|
|
|
|
// for each group
|
|
for(ChildAssociationRef group : groups)
|
|
{
|
|
NodeRef groupNode = group.getChildRef();
|
|
ret.addAll(getTransferTargets(groupNode));
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* Get all transfer targets in the specified group
|
|
*/
|
|
public Set<TransferTarget> getTransferTargets(String groupName)
|
|
{
|
|
NodeRef home = getTransferHome();
|
|
|
|
// get group with assoc groupName
|
|
NodeRef groupNode = nodeService.getChildByName(home, ContentModel.ASSOC_CONTAINS, groupName);
|
|
|
|
if(groupNode == null)
|
|
{
|
|
// No transfer group.
|
|
throw new TransferException(MSG_NO_GROUP, new Object[]{groupName});
|
|
}
|
|
|
|
return getTransferTargets(groupNode);
|
|
}
|
|
|
|
/**
|
|
* Given the noderef of a group of transfer targets, return all the contained transfer targets.
|
|
* @param groupNode
|
|
* @return
|
|
*/
|
|
private Set<TransferTarget> getTransferTargets(NodeRef groupNode)
|
|
{
|
|
Set<TransferTarget> result = new HashSet<TransferTarget>();
|
|
List<ChildAssociationRef>children = nodeService.getChildAssocs(groupNode, ContentModel.ASSOC_CONTAINS, RegexQNamePattern.MATCH_ALL);
|
|
|
|
for(ChildAssociationRef child : children)
|
|
{
|
|
if(nodeService.getType(child.getChildRef()).equals(TransferModel.TYPE_TRANSFER_TARGET))
|
|
{
|
|
TransferTargetImpl newTarget = new TransferTargetImpl();
|
|
mapTransferTarget(child.getChildRef(), newTarget);
|
|
result.add(newTarget);
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
*
|
|
*/
|
|
public void deleteTransferTarget(String name)
|
|
{
|
|
NodeRef nodeRef = lookupTransferTarget(name);
|
|
|
|
if(nodeRef == null)
|
|
{
|
|
// target does not exist
|
|
throw new TransferException(MSG_NO_TARGET, new Object[]{name} );
|
|
}
|
|
nodeService.deleteNode(nodeRef);
|
|
}
|
|
|
|
/**
|
|
* Enables/Disables the named transfer target
|
|
*/
|
|
public void enableTransferTarget(String name, boolean enable)
|
|
{
|
|
NodeRef nodeRef = lookupTransferTarget(name);
|
|
nodeService.setProperty(nodeRef, TransferModel.PROP_ENABLED, new Boolean(enable));
|
|
}
|
|
|
|
public boolean targetExists(String name)
|
|
{
|
|
return (lookupTransferTarget(name) != null);
|
|
}
|
|
|
|
/**
|
|
*
|
|
*/
|
|
public TransferTarget getTransferTarget(String name)
|
|
{
|
|
NodeRef nodeRef = lookupTransferTarget(name);
|
|
|
|
if(nodeRef == null)
|
|
{
|
|
// target does not exist
|
|
throw new TransferException(MSG_NO_TARGET, new Object[]{name} );
|
|
}
|
|
TransferTargetImpl newTarget = new TransferTargetImpl();
|
|
mapTransferTarget(nodeRef, newTarget);
|
|
|
|
return newTarget;
|
|
}
|
|
|
|
/**
|
|
* create or update a transfer target.
|
|
*/
|
|
public TransferTarget saveTransferTarget(TransferTarget update)
|
|
{
|
|
if(update.getNodeRef() == null)
|
|
{
|
|
// This is a save for the first time
|
|
return createTransferTarget(update);
|
|
}
|
|
|
|
NodeRef nodeRef = lookupTransferTarget(update.getName());
|
|
if(nodeRef == null)
|
|
{
|
|
// target does not exist
|
|
throw new TransferException(MSG_NO_TARGET, new Object[]{update.getName()} );
|
|
}
|
|
|
|
Map<QName, Serializable> properties = new HashMap<QName, Serializable>();
|
|
properties.put(TransferModel.PROP_ENDPOINT_HOST, update.getEndpointHost());
|
|
properties.put(TransferModel.PROP_ENDPOINT_PORT, update.getEndpointPort());
|
|
properties.put(TransferModel.PROP_ENDPOINT_PROTOCOL, update.getEndpointProtocol());
|
|
properties.put(TransferModel.PROP_ENDPOINT_PATH, update.getEndpointPath());
|
|
properties.put(TransferModel.PROP_USERNAME, update.getUsername());
|
|
properties.put(TransferModel.PROP_PASSWORD, new String(encrypt(update.getPassword())));
|
|
|
|
// titled aspect
|
|
properties.put(ContentModel.PROP_TITLE, update.getTitle());
|
|
properties.put(ContentModel.PROP_NAME, update.getName());
|
|
properties.put(ContentModel.PROP_DESCRIPTION, update.getDescription());
|
|
|
|
properties.put(TransferModel.PROP_ENABLED, new Boolean(update.isEnabled()));
|
|
nodeService.setProperties(nodeRef, properties);
|
|
|
|
TransferTargetImpl newTarget = new TransferTargetImpl();
|
|
mapTransferTarget(nodeRef, newTarget);
|
|
return newTarget;
|
|
}
|
|
|
|
/**
|
|
* Transfer async.
|
|
*
|
|
* @param targetName
|
|
* @param definition
|
|
* @param callbacks
|
|
*
|
|
*/
|
|
public void transferAsync(String targetName, TransferDefinition definition, TransferCallback... callbacks)
|
|
{
|
|
transferAsync(targetName, definition, Arrays.asList(callbacks));
|
|
}
|
|
|
|
/**
|
|
* Transfer async.
|
|
*
|
|
* @param targetName
|
|
* @param definition
|
|
* @param callbacks
|
|
*
|
|
*/
|
|
public void transferAsync(String targetName, TransferDefinition definition, Collection<TransferCallback> callbacks)
|
|
{
|
|
/**
|
|
* Event processor for this transfer instance
|
|
*/
|
|
final TransferEventProcessor eventProcessor = new TransferEventProcessor();
|
|
if(callbacks != null)
|
|
{
|
|
eventProcessor.observers.addAll(callbacks);
|
|
}
|
|
|
|
/*
|
|
* Note:
|
|
* callback should be Serializable to be passed through the action API
|
|
* However Serializable is not used so it does not matter. Perhaps the action API should be
|
|
* changed? Or we could add a Serializable proxy here.
|
|
*/
|
|
|
|
Map<String, Serializable> params = new HashMap<String, Serializable>();
|
|
params.put("targetName", targetName);
|
|
params.put("definition", definition);
|
|
params.put("callbacks", (Serializable)callbacks);
|
|
|
|
Action transferAction = actionService.createAction("transfer-async", params);
|
|
|
|
/**
|
|
* Execute transfer async in its own transaction.
|
|
* The action service only runs actions in the post commit which is why there's
|
|
* a separate transaction here.
|
|
*/
|
|
boolean success = false;
|
|
UserTransaction trx = transactionService.getNonPropagatingUserTransaction();
|
|
try
|
|
{
|
|
trx.begin();
|
|
logger.debug("calling action service to execute action");
|
|
actionService.executeAction(transferAction, null, false, true);
|
|
trx.commit();
|
|
logger.debug("committed successfully");
|
|
success = true;
|
|
}
|
|
catch (Exception error)
|
|
{
|
|
logger.error("unexpected exception", error);
|
|
throw new AlfrescoRuntimeException(MSG_ERR_TRANSFER_ASYNC, error);
|
|
}
|
|
finally
|
|
{
|
|
if(!success)
|
|
{
|
|
try
|
|
{
|
|
logger.debug("rolling back after error");
|
|
trx.rollback();
|
|
}
|
|
catch (Exception error)
|
|
{
|
|
logger.error("unexpected exception during rollback", error);
|
|
// There's nothing much we can do here
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Transfer Synchronous
|
|
*
|
|
* @param targetName
|
|
* @param definition
|
|
* @param callbacks
|
|
*/
|
|
public TransferEndEvent transfer(String targetName, TransferDefinition definition, TransferCallback... callbacks)
|
|
throws TransferFailureException
|
|
{
|
|
return transfer(targetName, definition, Arrays.asList(callbacks));
|
|
}
|
|
|
|
/**
|
|
* Transfer Synchronous
|
|
*
|
|
* @param targetName
|
|
* @param definition
|
|
* @param callbacks
|
|
*/
|
|
public TransferEndEvent transfer(String targetName, TransferDefinition definition, Collection<TransferCallback> callbacks)
|
|
throws TransferFailureException
|
|
{
|
|
/**
|
|
* Event processor for this transfer instance
|
|
*/
|
|
final TransferEventProcessor eventProcessor = new TransferEventProcessor();
|
|
if(callbacks != null)
|
|
{
|
|
eventProcessor.observers.addAll(callbacks);
|
|
}
|
|
|
|
/**
|
|
* Now go ahead and do the transfer
|
|
*/
|
|
return transferImpl(targetName, definition, eventProcessor);
|
|
}
|
|
|
|
private TransferEndEvent transferImpl(String targetName, final TransferDefinition definition, final TransferEventProcessor eventProcessor)
|
|
throws TransferFailureException
|
|
{
|
|
if(logger.isDebugEnabled())
|
|
{
|
|
logger.debug("transfer started to :" + targetName);
|
|
}
|
|
|
|
// transfer end event
|
|
TransferEndEvent endEvent = null;
|
|
Exception failureException = null;
|
|
TransferTarget target = null;
|
|
Transfer transfer = null;
|
|
final List<TransferEvent> transferReportEvents = new LinkedList<TransferEvent>();
|
|
NodeRef sourceReport = null;
|
|
NodeRef destinationReport = null;
|
|
File manifest = null;
|
|
File requisite = null;
|
|
int pollRetries = 0;
|
|
int pollPosition = -1;
|
|
boolean cancelled = false;
|
|
|
|
|
|
Descriptor currentDescriptor = descriptorService.getCurrentRepositoryDescriptor();
|
|
Descriptor serverDescriptor = descriptorService.getServerDescriptor();
|
|
final String localRepositoryId = currentDescriptor.getId();
|
|
TransferVersion fromVersion = new TransferVersionImpl(serverDescriptor);
|
|
|
|
// Wire in the transferReport - so any callbacks are stored in transferReport
|
|
TransferCallback reportCallback = new TransferCallback()
|
|
{
|
|
public void processEvent(TransferEvent event)
|
|
{
|
|
transferReportEvents.add(event);
|
|
}
|
|
};
|
|
eventProcessor.addObserver(reportCallback);
|
|
|
|
TransferContext transferContext = new TransferContext();
|
|
|
|
// start transfer
|
|
ClientTransferState clientState = ClientTransferState.Begin;
|
|
while(clientState != ClientTransferState.Exit)
|
|
{
|
|
try
|
|
{
|
|
switch (clientState)
|
|
{
|
|
case Begin:
|
|
{
|
|
eventProcessor.start();
|
|
|
|
manifest = createManifest(definition, localRepositoryId, fromVersion, transferContext);
|
|
logger.debug("transfer begin");
|
|
target = getTransferTarget(targetName);
|
|
checkTargetEnabled(target);
|
|
transfer = transmitter.begin(target, localRepositoryId, fromVersion);
|
|
String transferId = transfer.getTransferId();
|
|
TransferStatus status = new TransferStatus();
|
|
transferMonitoring.put(transferId, status);
|
|
logger.debug("transfer begun transferId:" + transferId);
|
|
eventProcessor.begin(transferId);
|
|
checkCancel(transferId);
|
|
|
|
// next state
|
|
clientState = ClientTransferState.Prepare;
|
|
break;
|
|
}
|
|
|
|
case Prepare:
|
|
{
|
|
// check alfresco versions are compatible
|
|
TransferVersion toVersion = transfer.getToVersion();
|
|
if(!this.transferVersionChecker.checkTransferVersions(fromVersion, toVersion))
|
|
{
|
|
throw new TransferException(MSG_INCOMPATIBLE_VERSIONS, new Object[] {transfer.getTransferId(), fromVersion, toVersion});
|
|
}
|
|
|
|
// send Manifest, get the requsite back.
|
|
eventProcessor.sendSnapshot(1,1);
|
|
|
|
requisite = createRequisiteFile();
|
|
FileOutputStream reqOutput = new FileOutputStream(requisite);
|
|
transmitter.sendManifest(transfer, manifest, reqOutput);
|
|
logger.debug("manifest sent");
|
|
checkCancel(transfer.getTransferId());
|
|
|
|
if(logger.isDebugEnabled())
|
|
{
|
|
logger.debug("requisite file written to local filesystem");
|
|
try
|
|
{
|
|
outputFile(requisite);
|
|
}
|
|
catch (IOException error)
|
|
{
|
|
// This is debug code - so an exception thrown while debugging
|
|
logger.debug("error while outputting snapshotFile");
|
|
error.printStackTrace();
|
|
}
|
|
}
|
|
|
|
sendContent(transfer, definition, eventProcessor, manifest, requisite);
|
|
logger.debug("content sending finished");
|
|
checkCancel(transfer.getTransferId());
|
|
|
|
// prepare
|
|
eventProcessor.prepare();
|
|
transmitter.prepare(transfer);
|
|
checkCancel(transfer.getTransferId());
|
|
|
|
// next state
|
|
clientState = ClientTransferState.Commit;
|
|
break;
|
|
}
|
|
|
|
case Commit:
|
|
{
|
|
logger.debug("about to start committing transferId:" + transfer.getTransferId());
|
|
eventProcessor.commit();
|
|
transmitter.commit(transfer);
|
|
|
|
logger.debug("committing transferId:" + transfer.getTransferId());
|
|
checkCancel(transfer.getTransferId());
|
|
|
|
// next state
|
|
clientState = ClientTransferState.Poll;
|
|
break;
|
|
}
|
|
|
|
case Poll:
|
|
{
|
|
TransferProgress progress = null;
|
|
try
|
|
{
|
|
progress = transmitter.getStatus(transfer);
|
|
|
|
// reset retries for next poll
|
|
pollRetries = 0;
|
|
}
|
|
catch(TransferException e)
|
|
{
|
|
pollRetries++;
|
|
if (pollRetries == 3)
|
|
{
|
|
throw new TransferException(MSG_FAILED_TO_GET_TRANSFER_STATUS, new Object[] {target.getName()});
|
|
}
|
|
}
|
|
|
|
// check status
|
|
if (progress.getStatus() == TransferProgress.Status.ERROR)
|
|
{
|
|
Throwable targetError = progress.getError();
|
|
// NOTE: it's possible the error is not returned from pre v3.4 target repositories
|
|
if (targetError == null)
|
|
{
|
|
targetError = new TransferException(MSG_UNKNOWN_TARGET_ERROR);
|
|
}
|
|
if (Exception.class.isAssignableFrom(targetError.getClass()))
|
|
{
|
|
failureException = (Exception)targetError;
|
|
}
|
|
else
|
|
{
|
|
failureException = new TransferException(MSG_TARGET_ERROR, new Object[] {targetError.getMessage()}, targetError);
|
|
}
|
|
clientState = ClientTransferState.Finished;
|
|
break;
|
|
}
|
|
else if (progress.getStatus() == TransferProgress.Status.CANCELLED)
|
|
{
|
|
cancelled = true;
|
|
clientState = ClientTransferState.Finished;
|
|
break;
|
|
}
|
|
|
|
// notify transfer progress
|
|
if (progress.getCurrentPosition() != pollPosition)
|
|
{
|
|
pollPosition = progress.getCurrentPosition();
|
|
logger.debug("committing :" + pollPosition);
|
|
eventProcessor.committing(progress.getEndPosition(), pollPosition);
|
|
}
|
|
|
|
if (progress.getStatus() == TransferProgress.Status.COMPLETE)
|
|
{
|
|
clientState = ClientTransferState.Finished;
|
|
break;
|
|
}
|
|
|
|
checkCancel(transfer.getTransferId());
|
|
|
|
// NOTE: stay in poll state...
|
|
// sleep before next poll
|
|
try
|
|
{
|
|
Thread.sleep(commitPollDelay);
|
|
}
|
|
catch (InterruptedException e)
|
|
{
|
|
// carry on
|
|
}
|
|
break;
|
|
}
|
|
|
|
case Cancel:
|
|
{
|
|
logger.debug("Abort - waiting for target confirmation of cancel");
|
|
transmitter.abort(transfer);
|
|
|
|
// next state... poll for confirmation of cancel from target
|
|
clientState = ClientTransferState.Poll;
|
|
break;
|
|
}
|
|
|
|
case Finished:
|
|
{
|
|
try
|
|
{
|
|
TransferEndEventImpl endEventImpl = null;
|
|
String reportName = null;
|
|
|
|
try
|
|
{
|
|
if (failureException != null)
|
|
{
|
|
logger.debug("TransferException - unable to transfer", failureException);
|
|
TransferEventError errorEvent = new TransferEventError();
|
|
errorEvent.setTransferState(TransferEvent.TransferState.ERROR);
|
|
errorEvent.setException(failureException);
|
|
errorEvent.setMessage(failureException.getMessage());
|
|
endEventImpl = errorEvent;
|
|
reportName = "error";
|
|
}
|
|
else if (cancelled)
|
|
{
|
|
endEventImpl = new TransferEventCancelled();
|
|
endEventImpl.setTransferState(TransferEvent.TransferState.CANCELLED);
|
|
endEventImpl.setMessage("cancelled");
|
|
reportName = "cancelled";
|
|
}
|
|
else
|
|
{
|
|
logger.debug("committed transferId:" + transfer.getTransferId());
|
|
endEventImpl = new TransferEventSuccess();
|
|
endEventImpl.setTransferState(TransferEvent.TransferState.SUCCESS);
|
|
endEventImpl.setMessage("success");
|
|
reportName = "success";
|
|
}
|
|
|
|
// manually add the terminal event to the transfer report event list
|
|
transferReportEvents.add(endEventImpl);
|
|
}
|
|
catch(Exception e)
|
|
{
|
|
// report this failure as last resort
|
|
failureException = e;
|
|
reportName = "error";
|
|
logger.warn("Exception - unable to notify end transfer state", e);
|
|
}
|
|
|
|
reportName += "_" + new SimpleDateFormat("yyyyMMddhhmmssSSS").format(new Date());
|
|
|
|
try
|
|
{
|
|
if(transfer != null)
|
|
{
|
|
logger.debug("now pull back the destination transfer report");
|
|
destinationReport = persistDestinationTransferReport(reportName, transfer, target);
|
|
if (destinationReport != null)
|
|
{
|
|
eventProcessor.writeReport(destinationReport, TransferEventReport.ReportType.DESTINATION, endEventImpl.getTransferState());
|
|
}
|
|
}
|
|
|
|
logger.debug("now persist the client side transfer report");
|
|
sourceReport = persistTransferReport(reportName, transfer, target, definition, transferReportEvents, manifest, failureException);
|
|
if (sourceReport != null)
|
|
{
|
|
eventProcessor.writeReport(sourceReport, TransferEventReport.ReportType.SOURCE, endEventImpl.getTransferState());
|
|
}
|
|
}
|
|
catch(Exception e)
|
|
{
|
|
logger.warn("Exception - unable to write transfer reports", e);
|
|
}
|
|
|
|
try
|
|
{
|
|
endEventImpl.setLast(true);
|
|
endEventImpl.setSourceReport(sourceReport);
|
|
endEventImpl.setDestinationReport(destinationReport);
|
|
endEvent = endEventImpl;
|
|
eventProcessor.end(endEvent);
|
|
}
|
|
catch(Exception e)
|
|
{
|
|
// report this failure as last resort
|
|
failureException = e;
|
|
logger.warn("Exception - unable to notify end transfer state", e);
|
|
}
|
|
}
|
|
finally
|
|
{
|
|
clientState = ClientTransferState.Exit;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
catch(TransferCancelledException e)
|
|
{
|
|
logger.debug("Interrupted by transfer cancel request from client");
|
|
clientState = ClientTransferState.Cancel;
|
|
}
|
|
catch(Exception e)
|
|
{
|
|
logger.debug("Exception - unable to transfer", e);
|
|
|
|
/**
|
|
* Save the first exception that we encounter.
|
|
*/
|
|
if(failureException == null)
|
|
{
|
|
failureException = e;
|
|
}
|
|
if (transfer != null && (clientState == ClientTransferState.Begin ||
|
|
clientState == ClientTransferState.Prepare ||
|
|
clientState == ClientTransferState.Commit))
|
|
{
|
|
// we must first inform the target repository that a client failure has occurred to allow it to
|
|
// clean up appropriately, too
|
|
clientState = ClientTransferState.Cancel;
|
|
}
|
|
else
|
|
{
|
|
clientState = ClientTransferState.Finished;
|
|
}
|
|
}
|
|
}
|
|
|
|
try
|
|
{
|
|
if (endEvent == null)
|
|
{
|
|
TransferEventError error = new TransferEventError();
|
|
error.setTransferState(TransferEvent.TransferState.ERROR);
|
|
TransferFailureException endException = new TransferFailureException(error);
|
|
error.setMessage(endException.getMessage());
|
|
error.setException(endException);
|
|
error.setSourceReport(sourceReport);
|
|
error.setDestinationReport(destinationReport);
|
|
error.setLast(true);
|
|
endEvent = error;
|
|
}
|
|
if (endEvent instanceof TransferEventError)
|
|
{
|
|
TransferEventError endError = (TransferEventError)endEvent;
|
|
throw new TransferFailureException(endError);
|
|
}
|
|
return endEvent;
|
|
}
|
|
finally
|
|
{
|
|
// clean up
|
|
if (transfer != null)
|
|
{
|
|
transferMonitoring.remove(transfer.getTransferId());
|
|
}
|
|
if(manifest != null)
|
|
{
|
|
manifest.delete();
|
|
logger.debug("manifest file deleted");
|
|
}
|
|
|
|
if(requisite != null)
|
|
{
|
|
requisite.delete();
|
|
logger.debug("requisite file deleted");
|
|
}
|
|
}
|
|
}
|
|
|
|
private File createManifest(TransferDefinition definition, String repositoryId, TransferVersion fromVersion, TransferContext transferContext)
|
|
throws IOException, SAXException
|
|
{
|
|
// which nodes to write to the snapshot
|
|
Set<NodeRef>nodes = definition.getNodes();
|
|
Set<NodeRef>nodesToRemove = definition.getNodesToRemove();
|
|
|
|
if((nodes == null || nodes.size() == 0) && (nodesToRemove == null || nodesToRemove.size() == 0))
|
|
{
|
|
logger.debug("no nodes to transfer");
|
|
throw new TransferException(MSG_NO_NODES);
|
|
}
|
|
|
|
//If a noderef exists in both the "nodes" set and the "nodesToRemove" set then the nodesToRemove wins. It is removed
|
|
//from the "nodes" set...
|
|
if (nodes != null && nodesToRemove != null)
|
|
{
|
|
nodes.removeAll(nodesToRemove);
|
|
}
|
|
|
|
int nodeCount = ((nodes == null) ? 0 : nodes.size()) + ((nodesToRemove == null) ? 0 : nodesToRemove.size());
|
|
|
|
/**
|
|
* create snapshot
|
|
*/
|
|
logger.debug("create snapshot");
|
|
|
|
// where to put snapshot ?
|
|
File tempDir = TempFileProvider.getLongLifeTempDir(FILE_DIRECTORY);
|
|
File snapshotFile = TempFileProvider.createTempFile("TRX-SNAP", FILE_SUFFIX, tempDir);
|
|
Writer snapshotWriter = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(snapshotFile), "UTF-8"));
|
|
|
|
// Write the manifest file
|
|
TransferManifestWriter formatter = new XMLTransferManifestWriter();
|
|
TransferManifestHeader header = new TransferManifestHeader();
|
|
header.setRepositoryId(repositoryId);
|
|
header.setTransferVersion(fromVersion);
|
|
header.setCreatedDate(new Date());
|
|
header.setNodeCount(nodeCount);
|
|
header.setSync(definition.isSync());
|
|
header.setReadOnly(definition.isReadOnly());
|
|
formatter.startTransferManifest(snapshotWriter);
|
|
formatter.writeTransferManifestHeader(header);
|
|
if (nodes != null)
|
|
{
|
|
for (NodeRef nodeRef : nodes)
|
|
{
|
|
TransferManifestNode node = transferManifestNodeFactory.createTransferManifestNode(nodeRef, definition, transferContext);
|
|
formatter.writeTransferManifestNode(node);
|
|
}
|
|
}
|
|
if (nodesToRemove != null)
|
|
{
|
|
for (NodeRef nodeRef : nodesToRemove)
|
|
{
|
|
TransferManifestNode node = transferManifestNodeFactory.createTransferManifestNode(nodeRef, definition, transferContext,
|
|
true);
|
|
formatter.writeTransferManifestNode(node);
|
|
}
|
|
}
|
|
formatter.endTransferManifest();
|
|
snapshotWriter.close();
|
|
|
|
logger.debug("snapshot file written to local filesystem");
|
|
// If we are debugging then write the file to stdout.
|
|
if(logger.isDebugEnabled())
|
|
{
|
|
try
|
|
{
|
|
outputFile(snapshotFile);
|
|
}
|
|
catch (IOException error)
|
|
{
|
|
// This is debug code - so an exception thrown while debugging
|
|
logger.debug("error while outputting snapshotFile");
|
|
error.printStackTrace();
|
|
}
|
|
}
|
|
|
|
return snapshotFile;
|
|
}
|
|
|
|
private File createRequisiteFile()
|
|
{
|
|
File tempDir = TempFileProvider.getLongLifeTempDir(FILE_DIRECTORY);
|
|
File reqFile = TempFileProvider.createTempFile("TRX-REQ", FILE_SUFFIX, tempDir);
|
|
return reqFile;
|
|
}
|
|
|
|
private void sendContent(final Transfer transfer, final TransferDefinition definition, final TransferEventProcessor eventProcessor,
|
|
File manifest, File requisite)
|
|
throws SAXException, ParserConfigurationException, IOException
|
|
{
|
|
SAXParserFactory saxParserFactory = SAXParserFactory.newInstance();
|
|
SAXParser parser;
|
|
parser = saxParserFactory.newSAXParser();
|
|
|
|
/**
|
|
* Parse the requisite file to generate the delta list
|
|
*/
|
|
DeltaListRequsiteProcessor reqProcessor = new DeltaListRequsiteProcessor();
|
|
XMLTransferRequsiteReader reqReader = new XMLTransferRequsiteReader(reqProcessor);
|
|
parser.parse(requisite, reqReader);
|
|
|
|
final DeltaList deltaList = reqProcessor.getDeltaList();
|
|
|
|
/**
|
|
* Parse the manifest file and transfer chunks over
|
|
*
|
|
* ManifestFile -> Manifest Processor -> Chunker -> Transmitter
|
|
*
|
|
* Step 1: Create a chunker and wire it up to the transmitter
|
|
*/
|
|
final ContentChunker chunker = new ContentChunkerImpl();
|
|
final Long removeNodesRange = Long.valueOf(definition.getNodesToRemove() != null ? definition.getNodesToRemove().size() : 0);
|
|
final Long nodesRange = Long.valueOf( definition.getNodes() != null ? definition.getNodes().size() : 0);
|
|
|
|
final Long fRange = removeNodesRange + nodesRange;
|
|
chunker.setHandler(
|
|
new ContentChunkProcessor(){
|
|
private long counter = 0;
|
|
public void processChunk(Set<ContentData> data)
|
|
{
|
|
checkCancel(transfer.getTransferId());
|
|
logger.debug("send chunk to transmitter");
|
|
for(ContentData file : data)
|
|
{
|
|
counter++;
|
|
eventProcessor.sendContent(file, fRange, counter);
|
|
}
|
|
transmitter.sendContent(transfer, data);
|
|
}
|
|
}
|
|
);
|
|
|
|
/**
|
|
* Step 2 : create a manifest processor and wire it up to the chunker
|
|
*/
|
|
TransferManifestProcessor processor = new TransferManifestProcessor()
|
|
{
|
|
public void processTransferManifestNode(TransferManifestNormalNode node)
|
|
{
|
|
Set<ContentData> data = TransferManifestNodeHelper.getContentData(node);
|
|
for(ContentData d : data)
|
|
{
|
|
checkCancel(transfer.getTransferId());
|
|
logger.debug("add content to chunker");
|
|
|
|
/**
|
|
* Check with the deltaList whether we need to send the content item
|
|
*/
|
|
if(deltaList != null)
|
|
{
|
|
String partName = TransferCommons.URLToPartName(d.getContentUrl());
|
|
if(deltaList.getRequiredParts().contains(partName))
|
|
{
|
|
logger.debug("content is required :" + d.getContentUrl());
|
|
chunker.addContent(d);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// No delta list - so send all content items
|
|
chunker.addContent(d);
|
|
}
|
|
}
|
|
}
|
|
|
|
public void processTransferManifiestHeader(TransferManifestHeader header){/* NO-OP */ }
|
|
public void startTransferManifest(){ /* NO-OP */ }
|
|
public void endTransferManifest(){ /* NO-OP */ }
|
|
public void processTransferManifestNode(TransferManifestDeletedNode node)
|
|
{ /* NO-OP */
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Step 3: wire up the manifest reader to a manifest processor
|
|
*/
|
|
|
|
XMLTransferManifestReader reader = new XMLTransferManifestReader(processor);
|
|
|
|
/**
|
|
* Step 4: start the magic - Give the manifest file to the manifest reader
|
|
*/
|
|
parser.parse(manifest, reader);
|
|
chunker.flush();
|
|
}
|
|
|
|
/**
|
|
* CancelAsync
|
|
*/
|
|
public void cancelAsync(String transferHandle)
|
|
{
|
|
TransferStatus status = transferMonitoring.get(transferHandle);
|
|
if(status != null)
|
|
{
|
|
logger.debug("canceling transfer :" + transferHandle);
|
|
status.cancelMe = true;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Check whether the specified transfer should be cancelled.
|
|
* @param transferHandle
|
|
* @throws TransferException - the transfer has been cancelled.
|
|
*/
|
|
private void checkCancel(String transferHandle) throws TransferException
|
|
{
|
|
TransferStatus status = transferMonitoring.get(transferHandle);
|
|
if(status != null)
|
|
{
|
|
if(!status.cancelInProgress && status.cancelMe)
|
|
{
|
|
status.cancelInProgress = true;
|
|
throw new TransferCancelledException();
|
|
}
|
|
}
|
|
}
|
|
|
|
private void checkTargetEnabled(TransferTarget target) throws TransferException
|
|
{
|
|
if(!target.isEnabled())
|
|
{
|
|
logger.debug("target is not enabled");
|
|
throw new TransferException(MSG_TARGET_NOT_ENABLED, new Object[] {target.getName()});
|
|
}
|
|
}
|
|
|
|
public void setNodeService(NodeService nodeService)
|
|
{
|
|
this.nodeService = nodeService;
|
|
}
|
|
|
|
public void setSearchService(SearchService searchService)
|
|
{
|
|
this.searchService = searchService;
|
|
}
|
|
|
|
public void setSingletonCache(SimpleCache<String, NodeRef> singletonCache)
|
|
{
|
|
this.singletonCache = singletonCache;
|
|
}
|
|
|
|
public void setTransferSpaceQuery(String transferSpaceQuery)
|
|
{
|
|
this.transferSpaceQuery = transferSpaceQuery;
|
|
}
|
|
|
|
public void setDefaultTransferGroup(String defaultGroup)
|
|
{
|
|
this.defaultTransferGroup = defaultGroup;
|
|
}
|
|
|
|
public TransferTransmitter getTransmitter()
|
|
{
|
|
return transmitter;
|
|
}
|
|
|
|
public void setTransmitter(TransferTransmitter transmitter)
|
|
{
|
|
this.transmitter = transmitter;
|
|
}
|
|
|
|
// note: cache is tenant-aware (if using TransctionalCache impl)
|
|
|
|
private SimpleCache<String, NodeRef> singletonCache; // eg. for transferHomeNodeRef
|
|
private final String KEY_TRANSFER_HOME_NODEREF = "key.transferServiceImpl2Home.noderef";
|
|
|
|
protected NodeRef getTransferHome()
|
|
{
|
|
NodeRef transferHome = singletonCache.get(KEY_TRANSFER_HOME_NODEREF);
|
|
if(transferHome == null)
|
|
{
|
|
String query = transferSpaceQuery;
|
|
List<NodeRef> refs = searchService.selectNodes(nodeService.getRootNode(StoreRef.STORE_REF_WORKSPACE_SPACESSTORE), query, null, namespaceService, false);
|
|
if (refs.size() == 0)
|
|
{
|
|
// No transfer home.
|
|
throw new TransferException(MSG_NO_HOME, new Object[] { query });
|
|
}
|
|
if (refs.size() != 0)
|
|
{
|
|
transferHome = refs.get(0);
|
|
singletonCache.put(KEY_TRANSFER_HOME_NODEREF, transferHome);
|
|
}
|
|
}
|
|
return transferHome;
|
|
}
|
|
|
|
private char[] encrypt(char[] text)
|
|
{
|
|
// placeholder dummy implementation - add an 'E' to the start
|
|
// String dummy = new String("E" + text);
|
|
// String dummy = new String(text);
|
|
// return dummy.toCharArray();
|
|
return text;
|
|
}
|
|
|
|
private char[] decrypt(char[] text)
|
|
{
|
|
// placeholder dummy implementation - strips off leading 'E'
|
|
// String dummy = new String(text);
|
|
return text;
|
|
//return dummy.substring(1).toCharArray();
|
|
}
|
|
|
|
/**
|
|
*
|
|
* @param name
|
|
* @return
|
|
*/
|
|
private NodeRef lookupTransferTarget(String name)
|
|
{
|
|
NodeRef defaultGroup = getDefaultGroup();
|
|
return nodeService.getChildByName(defaultGroup, ContentModel.ASSOC_CONTAINS, name);
|
|
}
|
|
|
|
private void mapTransferTarget(NodeRef nodeRef, TransferTargetImpl def)
|
|
{
|
|
def.setNodeRef(nodeRef);
|
|
Map<QName, Serializable> properties = nodeService.getProperties(nodeRef);
|
|
String name = (String)properties.get(ContentModel.PROP_NAME);
|
|
|
|
String endpointPath = (String)properties.get(TransferModel.PROP_ENDPOINT_PATH);
|
|
if (endpointPath == null)
|
|
throw new TransferException(MSG_MISSING_ENDPOINT_PATH, new Object[] {name});
|
|
def.setEndpointPath(endpointPath);
|
|
|
|
String endpointProtocol = (String)properties.get(TransferModel.PROP_ENDPOINT_PROTOCOL);
|
|
if (endpointProtocol == null)
|
|
throw new TransferException(MSG_MISSING_ENDPOINT_PROTOCOL, new Object[] {name});
|
|
def.setEndpointProtocol(endpointProtocol);
|
|
|
|
String endpointHost = (String)properties.get(TransferModel.PROP_ENDPOINT_HOST);
|
|
if (endpointHost== null)
|
|
throw new TransferException(MSG_MISSING_ENDPOINT_HOST, new Object[] {name});
|
|
def.setEndpointHost(endpointHost);
|
|
|
|
Integer endpointPort = (Integer)properties.get(TransferModel.PROP_ENDPOINT_PORT);
|
|
if (endpointPort == null)
|
|
throw new TransferException(MSG_MISSING_ENDPOINT_PORT, new Object[] {name});
|
|
def.setEndpointPort(endpointPort);
|
|
|
|
String username = (String)properties.get(TransferModel.PROP_USERNAME);
|
|
if (username == null)
|
|
throw new TransferException(MSG_MISSING_ENDPOINT_USERNAME, new Object[] {name});
|
|
def.setUsername(username);
|
|
|
|
Serializable passwordVal = properties.get(TransferModel.PROP_PASSWORD);
|
|
if (passwordVal == null)
|
|
throw new TransferException(MSG_MISSING_ENDPOINT_PASSWORD, new Object[] {name});
|
|
if(passwordVal.getClass().isArray())
|
|
{
|
|
def.setPassword(decrypt((char[])passwordVal));
|
|
}
|
|
if(passwordVal instanceof String)
|
|
{
|
|
String password = (String)passwordVal;
|
|
def.setPassword(decrypt(password.toCharArray()));
|
|
}
|
|
|
|
def.setName(name);
|
|
def.setTitle((String)properties.get(ContentModel.PROP_TITLE));
|
|
def.setDescription((String)properties.get(ContentModel.PROP_DESCRIPTION));
|
|
|
|
if(nodeService.hasAspect(nodeRef, TransferModel.ASPECT_ENABLEABLE))
|
|
{
|
|
def.setEnabled((Boolean)properties.get(TransferModel.PROP_ENABLED));
|
|
}
|
|
else
|
|
{
|
|
// If the enableable aspect is not present then we don't want transfer failing.
|
|
def.setEnabled(Boolean.TRUE);
|
|
}
|
|
}
|
|
|
|
/* (non-Javadoc)
|
|
* @see org.alfresco.service.cmr.transfer.TransferService#verify(org.alfresco.service.cmr.transfer.TransferTarget)
|
|
*/
|
|
public void verify(TransferTarget target) throws TransferException
|
|
{
|
|
transmitter.verifyTarget(target);
|
|
}
|
|
|
|
/**
|
|
* Utility to dump the contents of a file to the console
|
|
* @param file
|
|
*/
|
|
private static void outputFile(File file) throws IOException
|
|
{
|
|
BufferedReader reader = new BufferedReader(new InputStreamReader(new FileInputStream(file), "UTF-8"));
|
|
String s = reader.readLine();
|
|
while(s != null)
|
|
{
|
|
System.out.println(s);
|
|
s = reader.readLine();
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Success transfer report
|
|
*/
|
|
private NodeRef persistTransferReport(final String transferName, final Transfer transfer, final TransferTarget target, final TransferDefinition definition,
|
|
final List<TransferEvent> events, final File snapshotFile, final Exception exception)
|
|
{
|
|
// persist the transfer report in its own transaction so it cannot be rolled back
|
|
RetryingTransactionCallback<NodeRef> writeReportCallback = new RetryingTransactionCallback<NodeRef>()
|
|
{
|
|
@Override
|
|
public NodeRef execute() throws Throwable
|
|
{
|
|
logger.debug("transfer report starting");
|
|
NodeRef reportNode = null;
|
|
if (exception != null)
|
|
{
|
|
reportNode = transferReporter.createTransferReport(transferName, exception, target, definition, events, snapshotFile);
|
|
}
|
|
else
|
|
{
|
|
reportNode = transferReporter.createTransferReport(transferName, transfer, target, definition, events, snapshotFile);
|
|
}
|
|
logger.debug("transfer report done");
|
|
return reportNode;
|
|
}
|
|
};
|
|
NodeRef reportNode = transactionService.getRetryingTransactionHelper().doInTransaction(writeReportCallback, false, true);
|
|
return reportNode;
|
|
}
|
|
|
|
/**
|
|
* Destination Transfer report
|
|
* @return the node ref of the transfer report or null if there isn't one.
|
|
*/
|
|
private NodeRef persistDestinationTransferReport(final String transferName,
|
|
final Transfer transfer,
|
|
final TransferTarget target)
|
|
{
|
|
// in its own transaction so it cannot be rolled back
|
|
RetryingTransactionCallback<NodeRef> writeReportCallback = new RetryingTransactionCallback<NodeRef>()
|
|
{
|
|
@Override
|
|
public NodeRef execute() throws Throwable
|
|
{
|
|
File tempDir = TempFileProvider.getLongLifeTempDir(FILE_DIRECTORY);
|
|
File destReportFile = TempFileProvider.createTempFile("TRX-DREP", FILE_SUFFIX, tempDir);
|
|
FileOutputStream destReportOutput = new FileOutputStream(destReportFile);
|
|
transmitter.getTransferReport(transfer, destReportOutput);
|
|
logger.debug("transfer report (destination) starting");
|
|
|
|
NodeRef reportNode = transferReporter.writeDestinationReport(transferName, target, destReportFile);
|
|
logger.debug("transfer report (destination) done");
|
|
|
|
if(destReportFile != null)
|
|
{
|
|
destReportFile.delete();
|
|
}
|
|
logger.debug("destination report temp file deleted");
|
|
|
|
return reportNode;
|
|
}
|
|
};
|
|
try
|
|
{
|
|
NodeRef reportNode = transactionService.getRetryingTransactionHelper().doInTransaction(writeReportCallback, false, true);
|
|
return reportNode;
|
|
}
|
|
catch (Throwable e)
|
|
{
|
|
// there's nothing we can do here. - but we do not want the exception to propogate up.
|
|
logger.debug("unexpected error while obtaining destination transfer report", e);
|
|
return null;
|
|
}
|
|
}
|
|
|
|
public void setTransferManifestNodeFactory(TransferManifestNodeFactory transferManifestNodeFactory)
|
|
{
|
|
this.transferManifestNodeFactory = transferManifestNodeFactory;
|
|
}
|
|
|
|
public void setActionService(ActionService actionService)
|
|
{
|
|
this.actionService = actionService;
|
|
}
|
|
|
|
public void setTransactionService(TransactionService transactionService)
|
|
{
|
|
this.transactionService = transactionService;
|
|
}
|
|
|
|
public void setTransferReporter(TransferReporter transferReporter)
|
|
{
|
|
this.transferReporter = transferReporter;
|
|
}
|
|
|
|
public void setCommitPollDelay(long commitPollDelay)
|
|
{
|
|
this.commitPollDelay = commitPollDelay;
|
|
}
|
|
|
|
public void setDescriptorService(DescriptorService descriptorService)
|
|
{
|
|
this.descriptorService = descriptorService;
|
|
}
|
|
|
|
public void setTransferVersionChecker(TransferVersionChecker transferVersionChecker)
|
|
{
|
|
this.transferVersionChecker = transferVersionChecker;
|
|
}
|
|
|
|
public void setNamespaceService(NamespaceService namespaceService)
|
|
{
|
|
this.namespaceService = namespaceService;
|
|
}
|
|
|
|
private class TransferStatus
|
|
{
|
|
boolean cancelMe = false;
|
|
boolean cancelInProgress = false;
|
|
}
|
|
|
|
}
|