Dave Ward b58125d078 Merged V4.1-BUG-FIX to HEAD
42804: Merged BRANCHES/DEV/BELARUS/V4.1-BUG-FIX-2012_10_17 to BRANCHES/DEV/V4.1-BUG-FIX:
      42748: ALF-14200: Adding Invalid Aspects Via CMIS ATOM API Results in NullPointerException
   42810: Fix for ALF-15276 - sys:locale Attribute No Longer Available From jsnode
   42814: ALF-15276 - small improvement to remove duplicated data from response
   42824: ALF-15048: Merged PATCHES/V4.0.2 to V4.1-BUG-FIX
        42724: ALF-16048: CLONE - Version history doesn't go beyond two versions (0.1 and 0.2) when dragged and dropped via CIFS from Mac Lion OSx
        42739: ALF-16048: New files missing from previous check in
        42742: ALF-16048: Another missing file.
   42839: ALF-16417: Fix "Hybrid Sync - can retain invalid cloud tickets in a local cache"
      - retry once for invalid auth 
      - also externalise the implicit/default cache config
   42849: NodeDAO: Added new method to retrieve specific store ID
    - public Pair<Long, StoreRef> getStore(StoreRef storeRef);
   42857: Merged DEV to V4.1-BUG-FIX
      42821: ALF-13506 : WCMQS Example Application Caching Causes Changes to Inconsistently Appear on the Editorial Web Site
             Concurrency was improved for AssetImpl class.
             The returned values of the collections were made unmodifiable in the classes which implement Resource interface.
   42872: ALF-15601: "Performance issue using CMIS method getChildren() - gets version history"
   - avoids getting the version history (an expensive operation) if possible i.e. in the case of current version (live) nodes like for getChildren
   42900: Merged DEV to V4.1-BUG-FIX
      42734: ALF-15335 : 'external' authentication subsystem debug information too scarce
         Extended debug information in the authentication subsystem.


git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@42904 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
2012-10-21 18:09:03 +00:00

410 lines
15 KiB
Java

package org.alfresco.filesys.repo;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.nio.channels.FileChannel;
import java.util.List;
import org.alfresco.filesys.alfresco.ExtendedDiskInterface;
import org.alfresco.filesys.alfresco.RepositoryDiskInterface;
import org.alfresco.filesys.repo.rules.Command;
import org.alfresco.filesys.repo.rules.commands.CloseFileCommand;
import org.alfresco.filesys.repo.rules.commands.CompoundCommand;
import org.alfresco.filesys.repo.rules.commands.CopyContentCommand;
import org.alfresco.filesys.repo.rules.commands.CreateFileCommand;
import org.alfresco.filesys.repo.rules.commands.DeleteFileCommand;
import org.alfresco.filesys.repo.rules.commands.DoNothingCommand;
import org.alfresco.filesys.repo.rules.commands.MoveFileCommand;
import org.alfresco.filesys.repo.rules.commands.OpenFileCommand;
import org.alfresco.filesys.repo.rules.commands.ReduceQuotaCommand;
import org.alfresco.filesys.repo.rules.commands.RemoveNoContentFileOnError;
import org.alfresco.filesys.repo.rules.commands.RemoveTempFileCommand;
import org.alfresco.filesys.repo.rules.commands.RenameFileCommand;
import org.alfresco.filesys.repo.rules.commands.RestoreFileCommand;
import org.alfresco.filesys.repo.rules.commands.ReturnValueCommand;
import org.alfresco.jlan.server.SrvSession;
import org.alfresco.jlan.server.filesys.TreeConnection;
import org.alfresco.repo.transaction.AlfrescoTransactionSupport.TxnReadState;
import org.alfresco.repo.transaction.RetryingTransactionHelper;
import org.alfresco.repo.transaction.RetryingTransactionHelper.RetryingTransactionCallback;
import org.alfresco.service.transaction.TransactionService;
import org.alfresco.util.FileFilterMode;
import org.alfresco.util.FileFilterMode.Client;
import org.alfresco.util.PropertyCheck;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
/**
* Content Disk Driver Command Executor
* <p>
* Executes commands against the repository.
*/
public class CommandExecutorImpl implements CommandExecutor
{
private static Log logger = LogFactory.getLog(CommandExecutorImpl.class);
// Services go here.
private TransactionService transactionService;
private RepositoryDiskInterface repositoryDiskInterface;
private ExtendedDiskInterface diskInterface;
public void init()
{
PropertyCheck.mandatory(this, "transactionService", transactionService);
PropertyCheck.mandatory(this, "diskInterface", diskInterface);
PropertyCheck.mandatory(this, "repositoryDiskInterface", getRepositoryDiskInterface());
}
@Override
public Object execute(final SrvSession sess, final TreeConnection tree, final Command command) throws IOException
{
TxnReadState readState = command.getTransactionRequired();
Object ret = null;
// No transaction required.
if(readState == TxnReadState.TXN_NONE)
{
ret = executeInternal(sess, tree, command, null);
}
else
{
// Yes a transaction is required.
RetryingTransactionHelper helper = transactionService.getRetryingTransactionHelper();
boolean readOnly = readState == TxnReadState.TXN_READ_ONLY;
RetryingTransactionCallback<Object> cb = new RetryingTransactionCallback<Object>()
{
/**
* Perform a set of commands as a unit of transactional work.
*
* @return Return the result of the unit of work
* @throws Throwable This can be anything and will guarantee either a retry or a rollback
*/
public Object execute() throws IOException
{
try
{
return executeInternal(sess, tree, command, null);
}
catch (IOException e)
{
// Ensure original checked IOExceptions get propagated
throw new PropagatingException(e);
}
}
};
try
{
ret = helper.doInTransaction(cb, readOnly);
}
catch(PropagatingException pe)
{
if(command instanceof CompoundCommand)
{
if(logger.isDebugEnabled())
{
logger.debug("error executing command :command" + command, pe);
}
CompoundCommand c = (CompoundCommand)command;
// Error Callback Here ?
List<Command> commands = c.getPostErrorCommands();
if(commands != null)
{
for(Command c2 : commands)
{
try
{
executeInternal(sess, tree, c2, ret);
}
catch(Throwable t)
{
logger.warn("caught and ignored exception from error handler", t);
// Swallow exception from error handler.
}
}
}
}
// Unwrap checked exceptions
throw (IOException) pe.getCause();
}
}
/**
* execute post commit commands.
*/
if(command instanceof CompoundCommand)
{
logger.debug("post commit of compound command");
CompoundCommand c = (CompoundCommand)command;
List<Command> commands = c.getPostCommitCommands();
if(commands != null)
{
for(Command c2 : commands)
{
// TODO - what about exceptions from post commit?
executeInternal(sess, tree, c2, ret);
}
}
}
return ret;
}
private Client getClient(SrvSession srvSession)
{
String clientStr = srvSession.getServer().getProtocolName().toLowerCase();
if(clientStr.equals("cifs"))
{
return Client.cifs;
}
else if(clientStr.equals("nfs"))
{
return Client.nfs;
}
else if(clientStr.equals("ftp"))
{
return Client.ftp;
}
else
{
return null;
}
}
/**
* @param sess
* @param tree
* @param command
* @param result
* @return
* @throws IOException
*/
private Object executeInternal(SrvSession sess, TreeConnection tree, Command command, Object result) throws IOException
{
FileFilterMode.setClient(getClient(sess));
try
{
if(command instanceof CompoundCommand)
{
Object ret = null;
logger.debug("compound command received");
CompoundCommand x = (CompoundCommand)command;
for(Command compoundPart : x.getCommands())
{
logger.debug("running part of compound command");
Object val = executeInternal(sess, tree, compoundPart, result);
if(val != null)
{
// Return the value from the last command.
ret = val;
}
}
return ret;
}
else if(command instanceof CreateFileCommand)
{
logger.debug("create file command");
CreateFileCommand create = (CreateFileCommand)command;
return repositoryDiskInterface.createFile(create.getRootNode(), create.getPath(), create.getAllocationSize());
}
else if(command instanceof RestoreFileCommand)
{
logger.debug("restore file command");
RestoreFileCommand restore = (RestoreFileCommand)command;
return repositoryDiskInterface.restoreFile(sess, tree, restore.getRootNode(), restore.getPath(), restore.getAllocationSize(), restore.getOriginalNodeRef());
}
else if(command instanceof DeleteFileCommand)
{
logger.debug("delete file command");
DeleteFileCommand delete = (DeleteFileCommand)command;
return repositoryDiskInterface.deleteFile2(sess, tree, delete.getRootNode(), delete.getPath());
}
else if(command instanceof OpenFileCommand)
{
logger.debug("open file command");
OpenFileCommand o = (OpenFileCommand)command;
OpenFileMode mode = o.getMode();
return repositoryDiskInterface.openFile(sess, tree, o.getRootNodeRef(), o.getPath(), mode, o.isTruncate());
}
else if(command instanceof CloseFileCommand)
{
logger.debug("close file command");
CloseFileCommand c = (CloseFileCommand)command;
repositoryDiskInterface.closeFile(c.getRootNodeRef(), c.getPath(), c.getNetworkFile());
}
else if(command instanceof ReduceQuotaCommand)
{
logger.debug("reduceQuota file command");
ReduceQuotaCommand r = (ReduceQuotaCommand)command;
repositoryDiskInterface.reduceQuota(sess, tree, r.getNetworkFile());
}
else if(command instanceof RenameFileCommand)
{
logger.debug("rename command");
RenameFileCommand rename = (RenameFileCommand)command;
diskInterface.renameFile(sess, tree, rename.getFromPath(), rename.getToPath());
}
else if(command instanceof MoveFileCommand)
{
logger.debug("move command");
MoveFileCommand rename = (MoveFileCommand)command;
diskInterface.renameFile(sess, tree, rename.getFromPath(), rename.getToPath());
}
else if(command instanceof CopyContentCommand)
{
if(logger.isDebugEnabled())
{
logger.debug("Copy content command - copy content");
}
CopyContentCommand copy = (CopyContentCommand)command;
repositoryDiskInterface.copyContent(copy.getRootNode(), copy.getFromPath(), copy.getToPath());
}
else if(command instanceof DoNothingCommand)
{
if(logger.isDebugEnabled())
{
logger.debug("Do Nothing Command - doing nothing");
}
}
else if(command instanceof ResultCallback)
{
if(logger.isDebugEnabled())
{
logger.debug("Result Callback");
}
ResultCallback callback = (ResultCallback)command;
callback.execute(result);
}
else if(command instanceof RemoveTempFileCommand)
{
RemoveTempFileCommand r = (RemoveTempFileCommand)command;
if(logger.isDebugEnabled())
{
logger.debug("Remove Temp File:" + r.getNetworkFile());
}
File file = r.getNetworkFile().getFile();
boolean isDeleted = file.delete();
if(!isDeleted)
{
logger.debug("unable to delete temp file:" + r.getNetworkFile() + ", closed="+ r.getNetworkFile().isClosed());
/*
* Unable to delete temporary file
* Could be a bug with the file handle not being closed, but yourkit does not
* find anything awry.
* There are reported Windows JVM bugs such as 4715154 ...
*/
FileChannel outChan = null;
try
{
outChan = new FileOutputStream(file).getChannel();
outChan.truncate(0);
}
catch (IOException e)
{
logger.debug("unable to clean up file", e);
}
finally
{
if(outChan != null)
{
try
{
outChan.close();
}
catch(IOException e){}
}
}
}
}
else if(command instanceof ReturnValueCommand)
{
ReturnValueCommand r = (ReturnValueCommand)command;
if(logger.isDebugEnabled())
{
logger.debug("Return value");
}
return r.getReturnValue();
}
else if(command instanceof RemoveNoContentFileOnError)
{
RemoveNoContentFileOnError r = (RemoveNoContentFileOnError)command;
if(logger.isDebugEnabled())
{
logger.debug("Remove no content file on error");
}
repositoryDiskInterface.deleteEmptyFile(r.getRootNodeRef(), r.getPath());
}
}
finally
{
FileFilterMode.clearClient();
}
return null;
}
public void setTransactionService(TransactionService transactionService)
{
this.transactionService = transactionService;
}
public TransactionService getTransactionService()
{
return transactionService;
}
public void setRepositoryDiskInterface(RepositoryDiskInterface repositoryDiskInterface)
{
this.repositoryDiskInterface = repositoryDiskInterface;
}
public RepositoryDiskInterface getRepositoryDiskInterface()
{
return repositoryDiskInterface;
}
public void setDiskInterface(ExtendedDiskInterface diskInterface)
{
this.diskInterface = diskInterface;
}
public ExtendedDiskInterface getDiskInterface()
{
return diskInterface;
}
/**
* A wrapper for checked exceptions to be passed through the retrying transaction handler.
*/
protected static class PropagatingException extends RuntimeException
{
private static final long serialVersionUID = 1L;
/**
* @param cause
*/
public PropagatingException(Throwable cause)
{
super(cause);
}
}
}