mirror of
				https://github.com/Alfresco/alfresco-community-repo.git
				synced 2025-10-15 15:02:20 +00:00 
			
		
		
		
	git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@22877 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
		
			
				
	
	
		
			3473 lines
		
	
	
		
			111 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
			
		
		
	
	
			3473 lines
		
	
	
		
			111 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
| /*
 | |
|  * Copyright (C) 2005-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 <http://www.gnu.org/licenses/>. */
 | |
| 
 | |
| package org.alfresco.repo.avm;
 | |
| 
 | |
| import java.io.File;
 | |
| import java.io.InputStream;
 | |
| import java.io.OutputStream;
 | |
| import java.util.ArrayList;
 | |
| import java.util.Date;
 | |
| import java.util.HashMap;
 | |
| import java.util.List;
 | |
| import java.util.Map;
 | |
| import java.util.Set;
 | |
| import java.util.SortedMap;
 | |
| 
 | |
| import org.alfresco.model.WCMModel;
 | |
| import org.alfresco.repo.avm.util.AVMUtil;
 | |
| import org.alfresco.repo.content.ContentStore;
 | |
| import org.alfresco.repo.domain.PropertyValue;
 | |
| import org.alfresco.repo.domain.avm.AVMStoreEntity;
 | |
| import org.alfresco.repo.domain.permissions.Acl;
 | |
| import org.alfresco.repo.security.permissions.ACLCopyMode;
 | |
| import org.alfresco.repo.security.permissions.AccessDeniedException;
 | |
| import org.alfresco.repo.transaction.AlfrescoTransactionSupport;
 | |
| import org.alfresco.service.cmr.avm.AVMBadArgumentException;
 | |
| import org.alfresco.service.cmr.avm.AVMCycleException;
 | |
| import org.alfresco.service.cmr.avm.AVMException;
 | |
| import org.alfresco.service.cmr.avm.AVMExistsException;
 | |
| import org.alfresco.service.cmr.avm.AVMNodeDescriptor;
 | |
| import org.alfresco.service.cmr.avm.AVMNotFoundException;
 | |
| import org.alfresco.service.cmr.avm.AVMStoreDescriptor;
 | |
| import org.alfresco.service.cmr.avm.AVMWrongTypeException;
 | |
| import org.alfresco.service.cmr.avm.LayeringDescriptor;
 | |
| import org.alfresco.service.cmr.avm.VersionDescriptor;
 | |
| import org.alfresco.service.cmr.repository.ContentData;
 | |
| import org.alfresco.service.cmr.repository.ContentReader;
 | |
| import org.alfresco.service.cmr.repository.ContentWriter;
 | |
| import org.alfresco.service.cmr.security.AccessStatus;
 | |
| import org.alfresco.service.cmr.security.PermissionContext;
 | |
| import org.alfresco.service.cmr.security.PermissionService;
 | |
| import org.alfresco.service.namespace.QName;
 | |
| import org.alfresco.util.FileNameValidator;
 | |
| import org.alfresco.util.Pair;
 | |
| import org.apache.commons.logging.Log;
 | |
| import org.apache.commons.logging.LogFactory;
 | |
| 
 | |
| /**
 | |
|  * This or AVMStore are the implementors of the operations specified by AVMService.
 | |
|  * 
 | |
|  * @author britt
 | |
|  */
 | |
| public class AVMRepository
 | |
| {
 | |
|     /** The i18n'ized messages */
 | |
|     private static final String MSG_CYCLE_ON_CREATE = "avm.cycle.create";
 | |
|     private static final String MSG_CYCLE_ON_LOOKUP = "avm.cycle.lookup";
 | |
|     
 | |
|     private static Log fgLogger = LogFactory.getLog(AVMRepository.class);
 | |
| 
 | |
|     /**
 | |
|      * The single instance of AVMRepository.
 | |
|      */
 | |
|     private static AVMRepository fgInstance;
 | |
| 
 | |
|     /**
 | |
|      * The current lookup count.
 | |
|      */
 | |
|     private ThreadLocal<Integer> fLookupCount;
 | |
| 
 | |
|     /**
 | |
|      * Reference to the ContentStoreImpl
 | |
|      */
 | |
|     private ContentStore fContentStore;
 | |
| 
 | |
|     /**
 | |
|      * The Lookup Cache instance.
 | |
|      */
 | |
|     private LookupCache fLookupCache;
 | |
| 
 | |
|     private AVMStoreDAO fAVMStoreDAO;
 | |
| 
 | |
|     private AVMNodeDAO fAVMNodeDAO;
 | |
| 
 | |
|     private VersionRootDAO fVersionRootDAO;
 | |
| 
 | |
|     private VersionLayeredNodeEntryDAO fVersionLayeredNodeEntryDAO;
 | |
| 
 | |
|     private AVMStorePropertyDAO fAVMStorePropertyDAO;
 | |
| 
 | |
|     private ChildEntryDAO fChildEntryDAO;
 | |
| 
 | |
|     private PermissionService fPermissionService;
 | |
| 
 | |
|     // A bunch of TransactionListeners that do work for this.
 | |
| 
 | |
|     /**
 | |
|      * One for create store.
 | |
|      */
 | |
|     private CreateStoreTxnListener fCreateStoreTxnListener;
 | |
| 
 | |
|     /**
 | |
|      * One for purge store.
 | |
|      */
 | |
|     private PurgeStoreTxnListener fPurgeStoreTxnListener;
 | |
| 
 | |
|     /**
 | |
|      * One for create version.
 | |
|      */
 | |
|     private CreateVersionTxnListener fCreateVersionTxnListener;
 | |
| 
 | |
|     /**
 | |
|      * One for purge version.
 | |
|      */
 | |
|     private PurgeVersionTxnListener fPurgeVersionTxnListener;
 | |
| 
 | |
|     /**
 | |
|      * Create a new one.
 | |
|      */
 | |
|     public AVMRepository()
 | |
|     {
 | |
|         fLookupCount = new ThreadLocal<Integer>();
 | |
|         fgInstance = this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Set the ContentService.
 | |
|      */
 | |
|     public void setContentStore(ContentStore store)
 | |
|     {
 | |
|         fContentStore = store;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Set the Lookup Cache instance.
 | |
|      * 
 | |
|      * @param cache
 | |
|      *            The instance to set.
 | |
|      */
 | |
|     public void setLookupCache(LookupCache cache)
 | |
|     {
 | |
|         fLookupCache = cache;
 | |
|     }
 | |
| 
 | |
|     public void setCreateStoreTxnListener(CreateStoreTxnListener listener)
 | |
|     {
 | |
|         fCreateStoreTxnListener = listener;
 | |
|     }
 | |
| 
 | |
|     public void setPurgeStoreTxnListener(PurgeStoreTxnListener listener)
 | |
|     {
 | |
|         fPurgeStoreTxnListener = listener;
 | |
|     }
 | |
| 
 | |
|     public void setCreateVersionTxnListener(CreateVersionTxnListener listener)
 | |
|     {
 | |
|         fCreateVersionTxnListener = listener;
 | |
|     }
 | |
| 
 | |
|     public void setPurgeVersionTxnListener(PurgeVersionTxnListener listener)
 | |
|     {
 | |
|         fPurgeVersionTxnListener = listener;
 | |
|     }
 | |
| 
 | |
|     public void setAvmStoreDAO(AVMStoreDAO dao)
 | |
|     {
 | |
|         fAVMStoreDAO = dao;
 | |
|     }
 | |
| 
 | |
|     public void setAvmNodeDAO(AVMNodeDAO dao)
 | |
|     {
 | |
|         fAVMNodeDAO = dao;
 | |
|     }
 | |
| 
 | |
|     public void setVersionRootDAO(VersionRootDAO dao)
 | |
|     {
 | |
|         fVersionRootDAO = dao;
 | |
|     }
 | |
| 
 | |
|     public void setVersionLayeredNodeEntryDAO(VersionLayeredNodeEntryDAO dao)
 | |
|     {
 | |
|         fVersionLayeredNodeEntryDAO = dao;
 | |
|     }
 | |
| 
 | |
|     public void setAvmStorePropertyDAO(AVMStorePropertyDAO dao)
 | |
|     {
 | |
|         fAVMStorePropertyDAO = dao;
 | |
|     }
 | |
| 
 | |
|     public void setChildEntryDAO(ChildEntryDAO dao)
 | |
|     {
 | |
|         fChildEntryDAO = dao;
 | |
|     }
 | |
| 
 | |
|     public void setPermissionService(PermissionService service)
 | |
|     {
 | |
|         fPermissionService = service;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Create a file.
 | |
|      * 
 | |
|      * @param path
 | |
|      *            The path to the containing directory.
 | |
|      * @param name
 | |
|      *            The name for the new file.
 | |
|      */
 | |
|     public OutputStream createFile(String path, String name)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             OutputStream out = store.createFile(pathParts[1], name);
 | |
|             return out;
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Create a file with the given File as content.
 | |
|      * 
 | |
|      * @param path
 | |
|      *            The path to the containing directory.
 | |
|      * @param name
 | |
|      *            The name to give the file.
 | |
|      * @param data
 | |
|      *            The file contents.
 | |
|      */
 | |
|     public void createFile(String path, String name, File data, List<QName> aspects, Map<QName, PropertyValue> properties)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             store.createFile(pathParts[1], name, data, aspects, properties);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Create a new directory.
 | |
|      * 
 | |
|      * @param path
 | |
|      *            The path to the containing directory.
 | |
|      * @param name
 | |
|      *            The name to give the directory.
 | |
|      */
 | |
|     public void createDirectory(String path, String name, List<QName> aspects, Map<QName, PropertyValue> properties)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             store.createDirectory(pathParts[1], name, aspects, properties);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Create a new directory. This assumes that the parent is already copied and therefore should only be used with
 | |
|      * great care.
 | |
|      * 
 | |
|      * @param parent
 | |
|      *            The parent node.
 | |
|      * @param name
 | |
|      *            The name of the new directory.
 | |
|      * @return A descriptor for the newly created directory.
 | |
|      */
 | |
|     public AVMNodeDescriptor createDirectory(AVMNodeDescriptor parent, String name)
 | |
|     {
 | |
|         AVMNode node = fAVMNodeDAO.getByID(parent.getId());
 | |
|         if (node == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException(parent.getId() + " not found.");
 | |
|         }
 | |
|         if (!(node instanceof DirectoryNode))
 | |
|         {
 | |
|             throw new AVMWrongTypeException("Not a directory.");
 | |
|         }
 | |
|         if (!can(null, node, PermissionService.CREATE_CHILDREN, true))
 | |
|         {
 | |
|             throw new AccessDeniedException("Not allowed to write in: " + parent);
 | |
|         }
 | |
|         // We need the store to do anything so...
 | |
|         String[] pathParts = SplitPath(parent.getPath());
 | |
|         AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|         if (store == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Store not found.");
 | |
|         }
 | |
|         DirectoryNode dir = (DirectoryNode) node;
 | |
|         DirectoryNode child = null;
 | |
|         Long parentAcl = dir.getAcl() == null ? null : dir.getAcl().getId();
 | |
|         if (dir instanceof LayeredDirectoryNode)
 | |
|         {
 | |
|             // TODO - collapse save/update
 | |
|             child = new LayeredDirectoryNodeImpl((String) null, store, null, parentAcl, ACLCopyMode.INHERIT);
 | |
|             ((LayeredDirectoryNode) child).setPrimaryIndirection(false);
 | |
|             ((LayeredDirectoryNode) child).setLayerID(parent.getLayerID());
 | |
|             
 | |
|             child.copyACLs(dir, ACLCopyMode.INHERIT);
 | |
|             
 | |
|             AVMDAOs.Instance().fAVMNodeDAO.update(child);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             child = new PlainDirectoryNodeImpl(store);
 | |
|             
 | |
|             child.copyACLs(dir, ACLCopyMode.INHERIT);
 | |
|             
 | |
|             AVMDAOs.Instance().fAVMNodeDAO.save(child);
 | |
|         }
 | |
|         
 | |
|         dir.putChild(name, child);
 | |
|         
 | |
|         fLookupCache.onWrite(pathParts[0]);
 | |
|         AVMNodeDescriptor desc = child.getDescriptor(parent.getPath(), name, parent.getIndirection(), parent.getIndirectionVersion());
 | |
|         return desc;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Create a new layered directory.
 | |
|      * 
 | |
|      * @param srcPath
 | |
|      *            The target indirection for the new layered directory.
 | |
|      * @param dstPath
 | |
|      *            The path to the containing directory.
 | |
|      * @param name
 | |
|      *            The name for the new directory.
 | |
|      */
 | |
|     public void createLayeredDirectory(String srcPath, String dstPath, String name)
 | |
|     {
 | |
|         if (dstPath.indexOf(srcPath) == 0)
 | |
|         {
 | |
|             throw new AVMCycleException(MSG_CYCLE_ON_CREATE);
 | |
|         }
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(dstPath);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             store.createLayeredDirectory(srcPath, pathParts[1], name);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Create a new layered file.
 | |
|      * 
 | |
|      * @param srcPath
 | |
|      *            The target indirection for the new layered file.
 | |
|      * @param dstPath
 | |
|      *            The path to the containing directory.
 | |
|      * @param name
 | |
|      *            The name of the new layered file.
 | |
|      */
 | |
|     public void createLayeredFile(String srcPath, String dstPath, String name)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(dstPath);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             store.createLayeredFile(srcPath, pathParts[1], name);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Create a new AVMStore.
 | |
|      * 
 | |
|      * @param name
 | |
|      *            The name to give the new AVMStore.
 | |
|      */
 | |
|     public void createAVMStore(String name)
 | |
|     {
 | |
|         createAVMStore(name, null);
 | |
|     }
 | |
|     
 | |
|     public void createAVMStore(String name, Map<QName, PropertyValue> props)
 | |
|     {
 | |
|         AlfrescoTransactionSupport.bindListener(fCreateStoreTxnListener);
 | |
|         if (getAVMStoreByName(name) != null)
 | |
|         {
 | |
|             throw new AVMExistsException("AVMStore exists: " + name);
 | |
|         }
 | |
|         // Newing up the object causes it to be written to the db.
 | |
|         AVMStore rep = new AVMStoreImpl(this, name);
 | |
|         
 | |
|         // Special handling for AVMStore creation.
 | |
|         AVMNode rootNode = rep.getRoot();
 | |
|         rootNode.setStoreNew(null);
 | |
|         fAVMNodeDAO.update(rootNode);
 | |
|         
 | |
|         
 | |
|         if (props != null)
 | |
|         {
 | |
|             setStoreProperties(name, props);
 | |
|         }
 | |
|         
 | |
|         fCreateStoreTxnListener.storeCreated(name);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Create a new branch.
 | |
|      * 
 | |
|      * @param version
 | |
|      *            The version to branch off.
 | |
|      * @param srcPath
 | |
|      *            The path to make a branch from.
 | |
|      * @param dstPath
 | |
|      *            The containing directory.
 | |
|      * @param name
 | |
|      *            The name of the new branch.
 | |
|      */
 | |
|     public void createBranch(int version, String srcPath, String dstPath, String name)
 | |
|     {
 | |
|         if (dstPath.indexOf(srcPath) == 0)
 | |
|         {
 | |
|             throw new AVMCycleException(MSG_CYCLE_ON_CREATE);
 | |
|         }
 | |
|         // Lookup the src node.
 | |
|         fLookupCount.set(1);
 | |
|         String[] pathParts;
 | |
|         Lookup sPath;
 | |
|         List<VersionLayeredNodeEntry> layeredEntries = null;
 | |
|         try
 | |
|         {
 | |
|             pathParts = SplitPath(srcPath);
 | |
|             AVMStore srcRepo = getAVMStoreByName(pathParts[0]);
 | |
|             if (srcRepo == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             if (version < 0)
 | |
|             {
 | |
|                 fLookupCache.onSnapshot(pathParts[0]);
 | |
|                 version = srcRepo.createSnapshot("Branch Snapshot", null, new HashMap<String, Integer>()).get(pathParts[0]);
 | |
|             }
 | |
|             sPath = srcRepo.lookup(version, pathParts[1], false, false);
 | |
|             if (sPath == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Path not found.");
 | |
|             }
 | |
|             VersionRoot lastVersion = fVersionRootDAO.getByVersionID(srcRepo, version);
 | |
|             layeredEntries = fVersionLayeredNodeEntryDAO.get(lastVersion);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|         // Lookup the destination directory.
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             pathParts = SplitPath(dstPath);
 | |
|             AVMStore dstRepo = getAVMStoreByName(pathParts[0]);
 | |
|             if (dstRepo == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             Lookup dPath = dstRepo.lookupDirectory(-1, pathParts[1], true);
 | |
|             if (dPath == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Path not found.");
 | |
|             }
 | |
|             DirectoryNode dirNode = (DirectoryNode) dPath.getCurrentNode();
 | |
|             if (!can(dstRepo, dirNode, PermissionService.ADD_CHILDREN, dPath.getDirectlyContained()))
 | |
|             {
 | |
|                 throw new AccessDeniedException("Not permitted to add children: " + dstPath);
 | |
|             }
 | |
|             AVMNode srcNode = sPath.getCurrentNode();
 | |
|             AVMNode dstNode = null;
 | |
|             // We do different things depending on what kind of thing we're
 | |
|             // branching from. I'd be considerably happier if we disallowed
 | |
|             // certain scenarios, but Jon won't let me :P (bhp).
 | |
| 
 | |
|             Long inheritAcl = srcNode.getAcl() == null ? null : srcNode.getAcl().getId();
 | |
| 
 | |
|             if (srcNode.getType() == AVMNodeType.PLAIN_DIRECTORY)
 | |
|             {
 | |
|                 dstNode = new PlainDirectoryNodeImpl((PlainDirectoryNode) srcNode, dstRepo, inheritAcl, ACLCopyMode.INHERIT);
 | |
|             }
 | |
|             else if (srcNode.getType() == AVMNodeType.LAYERED_DIRECTORY)
 | |
|             {
 | |
|                 dstNode = new LayeredDirectoryNodeImpl((LayeredDirectoryNode) srcNode, dstRepo, sPath, false, inheritAcl, ACLCopyMode.INHERIT);
 | |
|                 
 | |
|                 // note: re-use generated node id as a layer id
 | |
|                 ((LayeredDirectoryNode) dstNode).setLayerID(dstNode.getId());
 | |
|                 
 | |
|                 AVMDAOs.Instance().fAVMNodeDAO.update(dstNode);
 | |
|             }
 | |
|             else if (srcNode.getType() == AVMNodeType.LAYERED_FILE)
 | |
|             {
 | |
|                 dstNode = new LayeredFileNodeImpl((LayeredFileNode) srcNode, dstRepo, inheritAcl, ACLCopyMode.INHERIT);
 | |
|             }
 | |
|             else
 | |
|             // This is a plain file.
 | |
|             {
 | |
|                 dstNode = new PlainFileNodeImpl((PlainFileNode) srcNode, dstRepo, inheritAcl, ACLCopyMode.INHERIT);
 | |
|             }
 | |
|             // dstNode.setVersionID(dstRepo.getNextVersionID());
 | |
|             dstNode.setAncestor(srcNode);
 | |
|             dirNode.putChild(name, dstNode);
 | |
|             // dirNode.updateModTime();
 | |
|             String beginingPath = AVMUtil.normalizePath(srcPath);
 | |
|             String finalPath = AVMUtil.extendAVMPath(dstPath, name);
 | |
|             finalPath = AVMUtil.normalizePath(finalPath);
 | |
|             VersionRoot latestVersion = fVersionRootDAO.getMaxVersion(dstRepo);
 | |
|             for (VersionLayeredNodeEntry entry : layeredEntries)
 | |
|             {
 | |
|                 String path = entry.getPath();
 | |
|                 if (!path.startsWith(srcPath))
 | |
|                 {
 | |
|                     continue;
 | |
|                 }
 | |
|                 String newPath = finalPath + path.substring(beginingPath.length());
 | |
|                 VersionLayeredNodeEntry newEntry = new VersionLayeredNodeEntryImpl(latestVersion, newPath);
 | |
|                 fVersionLayeredNodeEntryDAO.save(newEntry);
 | |
|             }
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get an output stream to a file.
 | |
|      * 
 | |
|      * @param path
 | |
|      *            The full path to the file.
 | |
|      * @return An OutputStream.
 | |
|      */
 | |
|     public OutputStream getOutputStream(String path)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             OutputStream out = store.getOutputStream(pathParts[1]);
 | |
|             return out;
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get a content reader from a file node.
 | |
|      * 
 | |
|      * @param version
 | |
|      *            The version of the file.
 | |
|      * @param path
 | |
|      *            The path to the file.
 | |
|      * @return A ContentReader.
 | |
|      */
 | |
|     public ContentReader getContentReader(int version, String path)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found: " + pathParts[0]);
 | |
|             }
 | |
|             return store.getContentReader(version, pathParts[1]);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get a ContentWriter to a file node.
 | |
|      * 
 | |
|      * @param path
 | |
|      *            The path to the file.
 | |
|      * @param update true if the property must be updated atomically when the content write
 | |
|      *      stream is closed (attaches a listener to the stream); false if the client code
 | |
|      *      will perform the updates itself.
 | |
|      * @return A ContentWriter.
 | |
|      */
 | |
|     public ContentWriter createContentWriter(String path, boolean update)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found: " + pathParts[0]);
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             ContentWriter writer = store.createContentWriter(pathParts[1], update);
 | |
|             return writer;
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Rename a node.
 | |
|      * 
 | |
|      * @param srcPath
 | |
|      *            Source containing directory.
 | |
|      * @param srcName
 | |
|      *            Source name.
 | |
|      * @param dstPath
 | |
|      *            Destination containing directory.
 | |
|      * @param dstName
 | |
|      *            Destination name.
 | |
|      */
 | |
|     public void rename(String srcPath, String srcName, String dstPath, String dstName)
 | |
|     {
 | |
|         String extDst = AVMUtil.extendAVMPath(dstPath, "");
 | |
|         String extSrc = AVMUtil.extendAVMPath(AVMUtil.extendAVMPath(srcPath, srcName), "");
 | |
|         if (extDst.indexOf(extSrc) == 0)
 | |
|         {
 | |
|             throw new AVMCycleException(MSG_CYCLE_ON_CREATE);
 | |
|         }
 | |
|         fLookupCount.set(1);
 | |
|         String[] pathParts;
 | |
|         Lookup sPath;
 | |
|         DirectoryNode srcDir;
 | |
|         AVMNode srcNode;
 | |
|         try
 | |
|         {
 | |
|             pathParts = SplitPath(srcPath);
 | |
|             AVMStore srcRepo = getAVMStoreByName(pathParts[0]);
 | |
|             if (srcRepo == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             sPath = srcRepo.lookupDirectory(-1, pathParts[1], true);
 | |
|             if (sPath == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Path not found.");
 | |
|             }
 | |
|             srcDir = (DirectoryNode) sPath.getCurrentNode();
 | |
|             
 | |
|             Pair<AVMNode, Boolean> temp = srcDir.lookupChild(sPath, srcName, false);
 | |
|             srcNode = (temp == null) ? null : temp.getFirst();
 | |
|             if (srcNode == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Not found: " + srcName);
 | |
|             }
 | |
|             if (!can(srcRepo, srcNode, PermissionService.DELETE_NODE, false))
 | |
|             {
 | |
|                 throw new AccessDeniedException("Not allowed to delete target: " + srcPath);
 | |
|             }
 | |
|             
 | |
|             fLookupCache.onDelete(pathParts[0]);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             pathParts = SplitPath(dstPath);
 | |
|             AVMStore dstRepo = getAVMStoreByName(pathParts[0]);
 | |
|             if (dstRepo == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             Lookup dPath = dstRepo.lookupDirectory(-1, pathParts[1], true);
 | |
|             if (dPath == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Path not found.");
 | |
|             }
 | |
|             DirectoryNode dstDir = (DirectoryNode) dPath.getCurrentNode();
 | |
|             if (!can(dstRepo, dstDir, PermissionService.ADD_CHILDREN, dPath.getDirectlyContained()))
 | |
|             {
 | |
|                 throw new AccessDeniedException("Not allowed to write: " + dstPath);
 | |
|             }
 | |
|             Pair<AVMNode, Boolean> temp = dstDir.lookupChild(dPath, dstName, true);
 | |
|             AVMNode child = (temp == null) ? null : temp.getFirst();
 | |
|             
 | |
|             boolean renameCase = false;
 | |
|             if (child != null && child.getType() != AVMNodeType.DELETED_NODE)
 | |
|             {
 | |
|                 String avmSrcPath = AVMUtil.extendAVMPath(srcPath, srcName);
 | |
|                 String avmDstPath = AVMUtil.extendAVMPath(dstPath, dstName);
 | |
|                 
 | |
|                 if ((avmSrcPath.equalsIgnoreCase(avmDstPath)) && (! srcName.equals(dstName)))
 | |
|                 {
 | |
|                     // specific rename 'case' only (within a store)
 | |
|                     if (fgLogger.isDebugEnabled())
 | |
|                     {
 | |
|                         fgLogger.debug("rename: only change case: from "+avmSrcPath+" to "+avmDstPath);
 | |
|                     }
 | |
|                     renameCase = true;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     throw new AVMExistsException("Node exists: " + dstName);
 | |
|                 }
 | |
|             }
 | |
|             
 | |
|             if (! renameCase)
 | |
|             {
 | |
|                 // general rename/move
 | |
|                 
 | |
|                 Long parentAcl = dstDir.getAcl() == null ? null : dstDir.getAcl().getId();
 | |
|                 
 | |
|                 AVMNode dstNode = null;
 | |
|                 // We've passed the check, so we can go ahead and do the rename.
 | |
|                 if (srcNode.getType() == AVMNodeType.PLAIN_DIRECTORY)
 | |
|                 {
 | |
|                     // If the source is layered then the renamed thing needs to be layered also.
 | |
|                     if (sPath.isLayered())
 | |
|                     {
 | |
|                         // If this is a rename happening in the same layer we make a new
 | |
|                         // OverlayedDirectoryNode that is not a primary indirection layer.
 | |
|                         // Otherwise we do make the new OverlayedDirectoryNode a primary
 | |
|                         // Indirection layer. This complexity begs the question of whether
 | |
|                         // we should allow renames from within one layer to within another
 | |
|                         // layer. Allowing it makes the logic absurdly complex.
 | |
|                         if (dPath.isLayered() && dPath.getTopLayer().equals(sPath.getTopLayer()))
 | |
|                         {
 | |
|                             dstNode = new LayeredDirectoryNodeImpl((PlainDirectoryNode) srcNode, dstRepo, sPath, true, parentAcl, ACLCopyMode.COPY);
 | |
|                             ((LayeredDirectoryNode) dstNode).setLayerID(sPath.getTopLayer().getLayerID());
 | |
|                         }
 | |
|                         else
 | |
|                         {
 | |
|                             dstNode = new LayeredDirectoryNodeImpl((DirectoryNode) srcNode, dstRepo, sPath, srcName, parentAcl, ACLCopyMode.COPY);
 | |
|                             
 | |
|                             // note: re-use generated node id as a layer id
 | |
|                             ((LayeredDirectoryNode) dstNode).setLayerID(dstNode.getId());
 | |
|                         }
 | |
|                         
 | |
|                         AVMDAOs.Instance().fAVMNodeDAO.update(dstNode);
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         dstNode = new PlainDirectoryNodeImpl((PlainDirectoryNode) srcNode, dstRepo, parentAcl, ACLCopyMode.COPY);
 | |
|                     }
 | |
|                 }
 | |
|                 else if (srcNode.getType() == AVMNodeType.LAYERED_DIRECTORY)
 | |
|                 {
 | |
|                     if (!sPath.isLayered() || (sPath.isInThisLayer() && srcDir.getType() == AVMNodeType.LAYERED_DIRECTORY && ((LayeredDirectoryNode) srcDir).directlyContains(srcNode)))
 | |
|                     {
 | |
|                         Lookup srcLookup = lookup(-1, srcPath + "/" + srcName, true);
 | |
|                         // Use the simple 'copy' constructor.
 | |
|                         dstNode = new LayeredDirectoryNodeImpl((LayeredDirectoryNode) srcNode, dstRepo, srcLookup, true, parentAcl, ACLCopyMode.COPY);
 | |
|                         ((LayeredDirectoryNode) dstNode).setLayerID(((LayeredDirectoryNode) srcNode).getLayerID());
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         // If the source node is a primary indirection, then the 'copy' constructor
 | |
|                         // is used. Otherwise the alternate constructor is called and its
 | |
|                         // indirection is calculated from it's source context.
 | |
|                         if (((LayeredDirectoryNode) srcNode).getPrimaryIndirection())
 | |
|                         {
 | |
|                             Lookup srcLookup = lookup(-1, srcPath + "/" + srcName, true);
 | |
|                             dstNode = new LayeredDirectoryNodeImpl((LayeredDirectoryNode) srcNode, dstRepo, srcLookup, true, parentAcl, ACLCopyMode.COPY);
 | |
|                         }
 | |
|                         else
 | |
|                         {
 | |
|                             dstNode = new LayeredDirectoryNodeImpl((DirectoryNode) srcNode, dstRepo, sPath, srcName, parentAcl, ACLCopyMode.COPY);
 | |
|                         }
 | |
|                         // What needs to be done here is dependent on whether the
 | |
|                         // rename is to a layered context. If so then it should get the layer id
 | |
|                         // of its destination parent. Otherwise it should get a new layer
 | |
|                         // id.
 | |
|                         if (dPath.isLayered())
 | |
|                         {
 | |
|                             ((LayeredDirectoryNode) dstNode).setLayerID(dPath.getTopLayer().getLayerID());
 | |
|                         }
 | |
|                         else
 | |
|                         {
 | |
|                             // note: re-use generated node id as a layer id
 | |
|                             ((LayeredDirectoryNode) dstNode).setLayerID(dstNode.getId());
 | |
|                         }
 | |
|                     }
 | |
|                     
 | |
|                     AVMDAOs.Instance().fAVMNodeDAO.update(dstNode);
 | |
|                 }
 | |
|                 else if (srcNode.getType() == AVMNodeType.LAYERED_FILE)
 | |
|                 {
 | |
|                     dstNode = new LayeredFileNodeImpl((LayeredFileNode) srcNode, dstRepo, parentAcl, ACLCopyMode.COPY);
 | |
|                 }
 | |
|                 else
 | |
|                 // This is a plain file node.
 | |
|                 {
 | |
|                     dstNode = new PlainFileNodeImpl((PlainFileNode) srcNode, dstRepo, parentAcl, ACLCopyMode.COPY);
 | |
|                 }
 | |
|                 
 | |
|                 srcDir.removeChild(sPath, srcName);
 | |
|                 // srcDir.updateModTime();
 | |
|                 // dstNode.setVersionID(dstRepo.getNextVersionID());
 | |
|                 if (child != null)
 | |
|                 {
 | |
|                     dstNode.setAncestor(child);
 | |
|                 }
 | |
|                 
 | |
|                 //dstDir.updateModTime();
 | |
|                 dstDir.putChild(dstName, dstNode);
 | |
|                 if (child == null)
 | |
|                 {
 | |
|                     dstNode.setAncestor(srcNode);
 | |
|                 }
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 // specific rename 'case' only (within a store)
 | |
|                 
 | |
|                 forceCopy(AVMUtil.extendAVMPath(srcPath, srcName));
 | |
|                 
 | |
|                 Pair<ChildEntry, Boolean> result = srcDir.lookupChildEntry(sPath, srcName, false);
 | |
|                 if (result != null)
 | |
|                 {
 | |
|                     ChildKey key = result.getFirst().getKey();
 | |
|                     key.setName(srcName);
 | |
|                     AVMDAOs.Instance().fChildEntryDAO.rename(key, dstName);
 | |
|                 }
 | |
|             }
 | |
|             
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Uncover a deleted name in a layered directory.
 | |
|      * 
 | |
|      * @param dirPath
 | |
|      *            The path to the layered directory.
 | |
|      * @param name
 | |
|      *            The name to uncover.
 | |
|      */
 | |
|     public void uncover(String dirPath, String name)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(dirPath);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             store.uncover(pathParts[1], name);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Create a snapshot of a single AVMStore.
 | |
|      * 
 | |
|      * @param store
 | |
|      *            The name of the repository.
 | |
|      * @param tag
 | |
|      *            The short description.
 | |
|      * @param description
 | |
|      *            The thick description.
 | |
|      * @return The version id of the newly snapshotted repository.
 | |
|      */
 | |
|     public Map<String, Integer> createSnapshot(String storeName, String tag, String description)
 | |
|     {
 | |
|         AlfrescoTransactionSupport.bindListener(fCreateVersionTxnListener);
 | |
|         AVMStore store = getAVMStoreByName(storeName);
 | |
|         if (store == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Store not found.");
 | |
|         }
 | |
|         Map<String, Integer> result = store.createSnapshot(tag, description, new HashMap<String, Integer>());
 | |
|         for (Map.Entry<String, Integer> entry : result.entrySet())
 | |
|         {
 | |
|             fLookupCache.onSnapshot(entry.getKey());
 | |
|             fCreateVersionTxnListener.versionCreated(entry.getKey(), entry.getValue());
 | |
|         }
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Remove a node and everything underneath it.
 | |
|      * 
 | |
|      * @param path
 | |
|      *            The path to the containing directory.
 | |
|      * @param name
 | |
|      *            The name of the node to remove.
 | |
|      */
 | |
|     public void remove(String path, String name)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             fLookupCache.onDelete(pathParts[0]);
 | |
|             store.removeNode(pathParts[1], name);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get rid of all content that lives only in the given AVMStore. Also removes the AVMStore.
 | |
|      * 
 | |
|      * @param name
 | |
|      *            The name of the AVMStore to purge.
 | |
|      */
 | |
|     public void purgeAVMStore(String name)
 | |
|     {
 | |
|         AlfrescoTransactionSupport.bindListener(fPurgeStoreTxnListener);
 | |
|         AVMStore store = getAVMStoreByName(name);
 | |
|         if (store == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Store not found.");
 | |
|         }
 | |
|         fLookupCache.onDelete(name);
 | |
|         AVMNode root = store.getRoot();
 | |
|         // TODO Probably a special PermissionService.PURGE is needed.
 | |
|         if (!can(store, root, PermissionService.DELETE_CHILDREN, true))
 | |
|         {
 | |
|             throw new AccessDeniedException("Not allowed to purge: " + name);
 | |
|         }
 | |
|         
 | |
|         root.setIsRoot(false);
 | |
|         fAVMNodeDAO.update(root);
 | |
|         
 | |
|         List<VersionRoot> vRoots = fVersionRootDAO.getAllInAVMStore(store);
 | |
|         for (VersionRoot vr : vRoots)
 | |
|         {
 | |
|             AVMNode node = fAVMNodeDAO.getByID(vr.getRoot().getId());
 | |
|             
 | |
|             node.setIsRoot(false);
 | |
|             fAVMNodeDAO.update(node);
 | |
|             
 | |
|             fVersionLayeredNodeEntryDAO.delete(vr);
 | |
|             fVersionRootDAO.delete(vr);
 | |
|         }
 | |
|         List<AVMNode> newGuys = fAVMNodeDAO.getNewInStore(store);
 | |
|         for (AVMNode newGuy : newGuys)
 | |
|         {
 | |
|             newGuy.setStoreNew(null);
 | |
|             
 | |
|             fAVMNodeDAO.update(newGuy);
 | |
|         }
 | |
|         fAVMStorePropertyDAO.delete(store);
 | |
|         fAVMStoreDAO.delete(store);
 | |
|         fAVMStoreDAO.invalidateCache();
 | |
|         fPurgeStoreTxnListener.storePurged(name);
 | |
|         
 | |
|         if (fgLogger.isDebugEnabled())
 | |
|         {
 | |
|             fgLogger.debug("Purged store: "+name);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Remove all content specific to a AVMRepository and version.
 | |
|      * 
 | |
|      * @param name
 | |
|      *            The name of the AVMStore.
 | |
|      * @param version
 | |
|      *            The version to purge.
 | |
|      */
 | |
|     public void purgeVersion(String name, int version)
 | |
|     {
 | |
|         AlfrescoTransactionSupport.bindListener(fPurgeVersionTxnListener);
 | |
|         AVMStore store = getAVMStoreByName(name);
 | |
|         if (store == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Store not found.");
 | |
|         }
 | |
|         fLookupCache.onDelete(name);
 | |
|         store.purgeVersion(version);
 | |
|         fPurgeVersionTxnListener.versionPurged(name, version);
 | |
|         
 | |
|         if (fgLogger.isDebugEnabled())
 | |
|         {
 | |
|             fgLogger.debug("Purged version: "+name+" "+version);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get an input stream from a file.
 | |
|      * 
 | |
|      * @param version
 | |
|      *            The version to look under.
 | |
|      * @param path
 | |
|      *            The path to the file.
 | |
|      * @return An InputStream.
 | |
|      */
 | |
|     public InputStream getInputStream(int version, String path)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             return store.getInputStream(version, pathParts[1]);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public InputStream getInputStream(AVMNodeDescriptor desc)
 | |
|     {
 | |
|         AVMNode node = fAVMNodeDAO.getByID(desc.getId());
 | |
|         if (!(node instanceof FileNode))
 | |
|         {
 | |
|             throw new AVMWrongTypeException(desc + " is not a File.");
 | |
|         }
 | |
|         if (!can(null, node, PermissionService.READ_CONTENT, false))
 | |
|         {
 | |
|             throw new AccessDeniedException("Not allowed to read content: " + desc);
 | |
|         }
 | |
|         FileNode file = (FileNode) node;
 | |
|         ContentData data = file.getContentData(null);
 | |
|         if (data == null)
 | |
|         {
 | |
|             throw new AVMException(desc + " has no content.");
 | |
|         }
 | |
|         ContentReader reader = fContentStore.getReader(data.getContentUrl());
 | |
|         return reader.getContentInputStream();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get a listing of a directory.
 | |
|      * 
 | |
|      * @param version
 | |
|      *            The version to look under.
 | |
|      * @param path
 | |
|      *            The path to the directory.
 | |
|      * @param includeDeleted
 | |
|      *            Whether to see DeletedNodes.
 | |
|      * @return A List of FolderEntries.
 | |
|      */
 | |
|     public SortedMap<String, AVMNodeDescriptor> getListing(int version, String path, boolean includeDeleted)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             return store.getListing(version, pathParts[1], includeDeleted);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the list of nodes directly contained in a directory.
 | |
|      * 
 | |
|      * @param version
 | |
|      *            The version to look under.
 | |
|      * @param path
 | |
|      *            The path to the directory to list.
 | |
|      * @return A Map of names to descriptors.
 | |
|      */
 | |
|     public SortedMap<String, AVMNodeDescriptor> getListingDirect(int version, String path, boolean includeDeleted)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             return store.getListingDirect(version, pathParts[1], includeDeleted);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the list of nodes directly contained in a directory.
 | |
|      * 
 | |
|      * @param dir
 | |
|      *            The descriptor to the directory node.
 | |
|      * @param includeDeleted
 | |
|      *            Whether to include deleted children.
 | |
|      * @return A Map of names to descriptors.
 | |
|      */
 | |
|     public SortedMap<String, AVMNodeDescriptor> getListingDirect(AVMNodeDescriptor dir, boolean includeDeleted)
 | |
|     {
 | |
|         AVMNode node = fAVMNodeDAO.getByID(dir.getId());
 | |
|         if (node == null)
 | |
|         {
 | |
|             throw new AVMBadArgumentException("Invalid Node.");
 | |
|         }
 | |
|         if (!can(null, node, PermissionService.READ_CHILDREN, false))
 | |
|         {
 | |
|             throw new AccessDeniedException("Not allowed to read children: " + dir);
 | |
|         }
 | |
|         if (node.getType() == AVMNodeType.PLAIN_DIRECTORY)
 | |
|         {
 | |
|             return getListing(dir, includeDeleted);
 | |
|         }
 | |
|         if (node.getType() != AVMNodeType.LAYERED_DIRECTORY)
 | |
|         {
 | |
|             throw new AVMWrongTypeException("Not a directory.");
 | |
|         }
 | |
|         LayeredDirectoryNode dirNode = (LayeredDirectoryNode) node;
 | |
|         return dirNode.getListingDirect(dir, includeDeleted);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get a directory listing from a directory node descriptor.
 | |
|      * 
 | |
|      * @param dir
 | |
|      *            The directory node descriptor.
 | |
|      * @return A SortedMap listing.
 | |
|      */
 | |
|     public SortedMap<String, AVMNodeDescriptor> getListing(AVMNodeDescriptor dir, boolean includeDeleted)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             AVMNode node = fAVMNodeDAO.getByID(dir.getId());
 | |
|             if (node == null)
 | |
|             {
 | |
|                 throw new AVMBadArgumentException("Invalid Node.");
 | |
|             }
 | |
|             if (node.getType() != AVMNodeType.LAYERED_DIRECTORY && node.getType() != AVMNodeType.PLAIN_DIRECTORY)
 | |
|             {
 | |
|                 throw new AVMWrongTypeException("Not a directory.");
 | |
|             }
 | |
|             if (!can(null, node, PermissionService.READ_CHILDREN, false))
 | |
|             {
 | |
|                 throw new AccessDeniedException("Not allowed to read children: " + dir);
 | |
|             }
 | |
|             DirectoryNode dirNode = (DirectoryNode) node;
 | |
|             SortedMap<String, AVMNodeDescriptor> listing = dirNode.getListing(dir, includeDeleted);
 | |
|             return listing;
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get a directory listing from a directory node descriptor fo children that match the given pattern
 | |
|      * 
 | |
|      * @param dir
 | |
|      *            The directory node descriptor.
 | |
|      * @return A SortedMap listing.
 | |
|      */
 | |
|     public SortedMap<String, AVMNodeDescriptor> getListing(AVMNodeDescriptor dir, String childNamePattern, boolean includeDeleted)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             AVMNode node = fAVMNodeDAO.getByID(dir.getId());
 | |
|             if (node == null)
 | |
|             {
 | |
|                 throw new AVMBadArgumentException("Invalid Node.");
 | |
|             }
 | |
|             if (node.getType() != AVMNodeType.LAYERED_DIRECTORY && node.getType() != AVMNodeType.PLAIN_DIRECTORY)
 | |
|             {
 | |
|                 throw new AVMWrongTypeException("Not a directory.");
 | |
|             }
 | |
|             if (!can(null, node, PermissionService.READ_CHILDREN, false))
 | |
|             {
 | |
|                 throw new AccessDeniedException("Not allowed to read children: " + dir);
 | |
|             }
 | |
|             DirectoryNode dirNode = (DirectoryNode) node;
 | |
|             SortedMap<String, AVMNodeDescriptor> listing = dirNode.getListing(dir, childNamePattern, includeDeleted);
 | |
|             return listing;
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     
 | |
|     /**
 | |
|      * Get the names of deleted nodes in a directory.
 | |
|      * 
 | |
|      * @param version
 | |
|      *            The version to look under.
 | |
|      * @param path
 | |
|      *            The path to the directory.
 | |
|      * @return A List of names.
 | |
|      */
 | |
|     public List<String> getDeleted(int version, String path)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             return store.getDeleted(version, pathParts[1]);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get descriptors of all AVMStores.
 | |
|      * 
 | |
|      * @return A list of all descriptors.
 | |
|      */
 | |
|     public List<AVMStoreDescriptor> getAVMStores()
 | |
|     {
 | |
|         List<AVMStoreEntity> storeEntities = AVMDAOs.Instance().newAVMStoreDAO.getAllStores();
 | |
|         
 | |
|         List<AVMStoreDescriptor> result = new ArrayList<AVMStoreDescriptor>(storeEntities.size());
 | |
|         for (AVMStoreEntity storeEntity : storeEntities)
 | |
|         {
 | |
|             AVMStoreImpl store = new AVMStoreImpl();
 | |
|             store.setId(storeEntity.getId());
 | |
|             store.setName(storeEntity.getName());
 | |
|             result.add(store.getDescriptor());
 | |
|         }
 | |
|         
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get a descriptor for an AVMStore.
 | |
|      * 
 | |
|      * @param name
 | |
|      *            The name to get.
 | |
|      * @return The descriptor.
 | |
|      */
 | |
|     public AVMStoreDescriptor getAVMStore(String name)
 | |
|     {
 | |
|         AVMStore store = getAVMStoreByName(name);
 | |
|         if (store == null)
 | |
|         {
 | |
|             return null;
 | |
|         }
 | |
|         return store.getDescriptor();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get all version for a given AVMStore.
 | |
|      * 
 | |
|      * @param name
 | |
|      *            The name of the AVMStore.
 | |
|      * @return A Set will all the version ids.
 | |
|      */
 | |
|     public List<VersionDescriptor> getAVMStoreVersions(String name)
 | |
|     {
 | |
|         AVMStore store = getAVMStoreByName(name);
 | |
|         if (store == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Store not found.");
 | |
|         }
 | |
|         return store.getVersions();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the set of versions between (inclusive) of the given dates. From or to may be null but not both.
 | |
|      * 
 | |
|      * @param name
 | |
|      *            The name of the AVMRepository.
 | |
|      * @param from
 | |
|      *            The earliest date.
 | |
|      * @param to
 | |
|      *            The latest date.
 | |
|      * @return The Set of version IDs.
 | |
|      */
 | |
|     public List<VersionDescriptor> getAVMStoreVersions(String name, Date from, Date to)
 | |
|     {
 | |
|         AVMStore store = getAVMStoreByName(name);
 | |
|         if (store == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Store not found.");
 | |
|         }
 | |
|         return store.getVersions(from, to);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the indirection path for a layered node.
 | |
|      * 
 | |
|      * @param version
 | |
|      *            The version to look under.
 | |
|      * @param path
 | |
|      *            The path to the node.
 | |
|      * @return The indirection path.
 | |
|      */
 | |
|     public String getIndirectionPath(int version, String path)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             return store.getIndirectionPath(version, pathParts[1]);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the next version id for the given AVMStore.
 | |
|      * 
 | |
|      * @param name
 | |
|      *            The name of the AVMStore.
 | |
|      * @return The next version id.
 | |
|      */
 | |
|     public int getLatestVersionID(String name)
 | |
|     {
 | |
|         AVMStore store = getAVMStoreByName(name);
 | |
|         if (store == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Store not found.");
 | |
|         }
 | |
|         return store.getNextVersionID();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the latest extant snapshotted version id.
 | |
|      * 
 | |
|      * @param name
 | |
|      *            The store name.
 | |
|      */
 | |
|     public int getLatestSnapshotID(String name)
 | |
|     {
 | |
|         AVMStore store = getAVMStoreByName(name);
 | |
|         if (store == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Store not found.");
 | |
|         }
 | |
|         return store.getLastVersionID();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get an AVMStore by name.
 | |
|      * 
 | |
|      * @param name
 | |
|      *            The name of the AVMStore.
 | |
|      * @return The AVMStore.
 | |
|      */
 | |
|     private AVMStore getAVMStoreByName(String name)
 | |
|     {
 | |
|         AVMStore store = fAVMStoreDAO.getByName(name);
 | |
|         return store;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get a descriptor for an AVMStore root.
 | |
|      * 
 | |
|      * @param version
 | |
|      *            The version to get.
 | |
|      * @param name
 | |
|      *            The name of the AVMStore.
 | |
|      * @return The descriptor for the root.
 | |
|      */
 | |
|     public AVMNodeDescriptor getAVMStoreRoot(int version, String name)
 | |
|     {
 | |
|         AVMStore store = getAVMStoreByName(name);
 | |
|         if (store == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Not found: " + name);
 | |
|         }
 | |
|         return store.getRoot(version);
 | |
|     }
 | |
| 
 | |
|     // TODO Fix this awful mess regarding cycle detection.
 | |
|     /**
 | |
|      * Lookup a node.
 | |
|      * 
 | |
|      * @param version
 | |
|      *            The version to look under.
 | |
|      * @param path
 | |
|      *            The path to lookup.
 | |
|      * @param includeDeleted
 | |
|      *            Whether to see DeletedNodes.
 | |
|      * @return A lookup object.
 | |
|      */
 | |
|     public Lookup lookup(int version, String path, boolean includeDeleted)
 | |
|     {
 | |
|         Integer count = fLookupCount.get();
 | |
|         try
 | |
|         {
 | |
|             if (count == null)
 | |
|             {
 | |
|                 fLookupCount.set(1);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 fLookupCount.set(count + 1);
 | |
|             }
 | |
|             if (fLookupCount.get() > 50)
 | |
|             {
 | |
|                 throw new AVMCycleException(MSG_CYCLE_ON_LOOKUP);
 | |
|             }
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 return null;
 | |
|             }
 | |
|             return store.lookup(version, pathParts[1], false, includeDeleted);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             if (count == null)
 | |
|             {
 | |
|                 fLookupCount.set(null);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Lookup a descriptor from a directory descriptor.
 | |
|      * 
 | |
|      * @param dir
 | |
|      *            The directory descriptor.
 | |
|      * @param name
 | |
|      *            The name of the child to lookup.
 | |
|      * @return The child's descriptor.
 | |
|      */
 | |
|     public AVMNodeDescriptor lookup(AVMNodeDescriptor dir, String name, boolean includeDeleted)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             AVMNode node = fAVMNodeDAO.getByID(dir.getId());
 | |
|             if (node == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Not found: " + dir.getId());
 | |
|             }
 | |
|             if (node.getType() != AVMNodeType.LAYERED_DIRECTORY && node.getType() != AVMNodeType.PLAIN_DIRECTORY)
 | |
|             {
 | |
|                 throw new AVMWrongTypeException("Not a directory.");
 | |
|             }
 | |
|             DirectoryNode dirNode = (DirectoryNode) node;
 | |
|             if (!can(null, dirNode, PermissionService.READ_CHILDREN, false))
 | |
|             {
 | |
|                 throw new AccessDeniedException("Not allowed to read children: " + dir);
 | |
|             }
 | |
|             return dirNode.lookupChild(dir, name, includeDeleted);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get all the paths to a particular node.
 | |
|      * 
 | |
|      * @param desc
 | |
|      *            The node descriptor.
 | |
|      * @return The list of version, paths.
 | |
|      */
 | |
|     public List<Pair<Integer, String>> getPaths(AVMNodeDescriptor desc)
 | |
|     {
 | |
|         AVMNode node = fAVMNodeDAO.getByID(desc.getId());
 | |
|         if (node == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Not found: " + desc);
 | |
|         }
 | |
|         List<Pair<Integer, String>> paths = new ArrayList<Pair<Integer, String>>();
 | |
|         List<String> components = new ArrayList<String>();
 | |
|         recursiveGetPaths(node, components, paths);
 | |
|         return paths;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get a single valid path for a node.
 | |
|      * 
 | |
|      * @param desc
 | |
|      *            The node descriptor.
 | |
|      * @return A version, path
 | |
|      */
 | |
|     public Pair<Integer, String> getAPath(AVMNodeDescriptor desc)
 | |
|     {
 | |
|         AVMNode node = fAVMNodeDAO.getByID(desc.getId());
 | |
|         if (node == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Could not find node: " + desc);
 | |
|         }
 | |
|         if (fgLogger.isDebugEnabled())
 | |
|         {
 | |
|             fgLogger.debug("Getting A Path for: " + desc);
 | |
|         }
 | |
|         List<String> components = new ArrayList<String>();
 | |
|         return recursiveGetAPath(node, components);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get all paths for a node reachable by HEAD.
 | |
|      * 
 | |
|      * @param desc
 | |
|      *            The node descriptor.
 | |
|      * @return A List of all the version, path Pairs that match.
 | |
|      */
 | |
|     public List<Pair<Integer, String>> getHeadPaths(AVMNodeDescriptor desc)
 | |
|     {
 | |
|         AVMNode node = fAVMNodeDAO.getByID(desc.getId());
 | |
|         if (node == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Not found: " + desc.getPath());
 | |
|         }
 | |
|         List<Pair<Integer, String>> paths = new ArrayList<Pair<Integer, String>>();
 | |
|         List<String> components = new ArrayList<String>();
 | |
|         recursiveGetHeadPaths(node, components, paths);
 | |
|         return paths;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Gets all the pass from to the given node starting from the give version root.
 | |
|      * 
 | |
|      * @param version
 | |
|      *            The version root.
 | |
|      * @param node
 | |
|      *            The node to get the paths of.
 | |
|      * @return A list of all paths in the given version to the node.
 | |
|      */
 | |
|     public List<String> getVersionPaths(VersionRoot version, AVMNode node)
 | |
|     {
 | |
|         List<String> paths = new ArrayList<String>();
 | |
|         List<String> components = new ArrayList<String>();
 | |
|         recursiveGetVersionPaths(node, components, paths, version.getRoot(), version.getAvmStore().getName());
 | |
|         return paths;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Helper to get all version paths.
 | |
|      * 
 | |
|      * @param node
 | |
|      *            The current node we are examining.
 | |
|      * @param components
 | |
|      *            The current path components.
 | |
|      * @param paths
 | |
|      *            The list to contain found paths.
 | |
|      * @param root
 | |
|      *            The root node of the version.
 | |
|      * @param storeName
 | |
|      *            The name of the store.
 | |
|      */
 | |
|     private void recursiveGetVersionPaths(AVMNode node, List<String> components, List<String> paths, DirectoryNode root, String storeName)
 | |
|     {
 | |
|         if (!can(null, node, PermissionService.READ_CHILDREN, false))
 | |
|         {
 | |
|             return;
 | |
|         }
 | |
|         if (node.equals(root))
 | |
|         {
 | |
|             paths.add(this.makePath(components, storeName));
 | |
|             return;
 | |
|         }
 | |
|         List<ChildEntry> entries = fChildEntryDAO.getByChild(node);
 | |
|         for (ChildEntry entry : entries)
 | |
|         {
 | |
|             String name = entry.getKey().getName();
 | |
|             components.add(name);
 | |
|             AVMNode parent = entry.getKey().getParent();
 | |
|             recursiveGetVersionPaths(parent, components, paths, root, storeName);
 | |
|             components.remove(components.size() - 1);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get all paths in a particular store in the head version for a particular node.
 | |
|      * 
 | |
|      * @param desc
 | |
|      *            The node descriptor.
 | |
|      * @param store
 | |
|      *            The name of the store.
 | |
|      * @return All matching paths.
 | |
|      */
 | |
|     public List<Pair<Integer, String>> getPathsInStoreHead(AVMNodeDescriptor desc, String store)
 | |
|     {
 | |
|         AVMStore st = getAVMStoreByName(store);
 | |
|         if (st == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Store not found: " + store);
 | |
|         }
 | |
|         AVMNode node = fAVMNodeDAO.getByID(desc.getId());
 | |
|         if (node == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Not found: " + desc);
 | |
|         }
 | |
|         List<Pair<Integer, String>> paths = new ArrayList<Pair<Integer, String>>();
 | |
|         List<String> components = new ArrayList<String>();
 | |
|         recursiveGetPathsInStoreHead(node, components, paths, st.getRoot(), store);
 | |
|         return paths;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Do the actual work.
 | |
|      * 
 | |
|      * @param node
 | |
|      *            The current node.
 | |
|      * @param components
 | |
|      *            The currently accumulated path components.
 | |
|      * @param paths
 | |
|      *            The list to put full paths in.
 | |
|      */
 | |
|     private void recursiveGetPaths(AVMNode node, List<String> components, List<Pair<Integer, String>> paths)
 | |
|     {
 | |
|         if (!can(null, node, PermissionService.READ_CHILDREN, false))
 | |
|         {
 | |
|             return;
 | |
|         }
 | |
|         if (node.getIsRoot())
 | |
|         {
 | |
|             AVMStore store = fAVMStoreDAO.getByRoot(node);
 | |
|             if (store != null)
 | |
|             {
 | |
|                 addPath(components, -1, store.getName(), paths);
 | |
|             }
 | |
|             VersionRoot vr = fVersionRootDAO.getByRoot(node);
 | |
|             if (vr != null)
 | |
|             {
 | |
|                 addPath(components, vr.getVersionID(), vr.getAvmStore().getName(), paths);
 | |
|             }
 | |
|             return;
 | |
|         }
 | |
|         List<ChildEntry> entries = fChildEntryDAO.getByChild(node);
 | |
|         for (ChildEntry entry : entries)
 | |
|         {
 | |
|             String name = entry.getKey().getName();
 | |
|             components.add(name);
 | |
|             AVMNode parent = entry.getKey().getParent();
 | |
|             recursiveGetPaths(parent, components, paths);
 | |
|             components.remove(components.size() - 1);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Do the work of getting one path for a node.
 | |
|      * 
 | |
|      * @param node
 | |
|      *            The node to get the path of.
 | |
|      * @param components
 | |
|      *            The storage for path components.
 | |
|      * @return A path or null.
 | |
|      */
 | |
|     private Pair<Integer, String> recursiveGetAPath(AVMNode node, List<String> components)
 | |
|     {
 | |
|         if (!can(null, node, PermissionService.READ_CHILDREN, false))
 | |
|         {
 | |
|             return null;
 | |
|         }
 | |
|         if (node.getIsRoot())
 | |
|         {
 | |
|             AVMStore store = fAVMStoreDAO.getByRoot(node);
 | |
|             if (store != null)
 | |
|             {
 | |
|                 if (fgLogger.isDebugEnabled())
 | |
|                 {
 | |
|                     fgLogger.debug("Found path in HEAD of: " + store.getName());
 | |
|                 }
 | |
|                 return new Pair<Integer, String>(-1, makePath(components, store.getName()));
 | |
|             }
 | |
|             VersionRoot vr = fVersionRootDAO.getByRoot(node);
 | |
|             if (vr != null)
 | |
|             {
 | |
|                 if (fgLogger.isDebugEnabled())
 | |
|                 {
 | |
|                     fgLogger.debug("Found path in version " + vr.getVersionID() + " in: " + vr.getAvmStore().getName());
 | |
|                 }
 | |
|                 return new Pair<Integer, String>(vr.getVersionID(), makePath(components, vr.getAvmStore().getName()));
 | |
|             }
 | |
|             return null;
 | |
|         }
 | |
|         List<ChildEntry> entries = fChildEntryDAO.getByChild(node);
 | |
|         for (ChildEntry entry : entries)
 | |
|         {
 | |
|             String name = entry.getKey().getName();
 | |
|             if (fgLogger.isDebugEnabled())
 | |
|             {
 | |
|                 fgLogger.debug("Found component: " + name);
 | |
|             }
 | |
|             components.add(name);
 | |
|             Pair<Integer, String> path = recursiveGetAPath(entry.getKey().getParent(), components);
 | |
|             if (path != null)
 | |
|             {
 | |
|                 return path;
 | |
|             }
 | |
|             components.remove(components.size() - 1);
 | |
|         }
 | |
|         return null;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Do the actual work.
 | |
|      * 
 | |
|      * @param node
 | |
|      *            The current node.
 | |
|      * @param components
 | |
|      *            The currently accumulated path components.
 | |
|      * @param paths
 | |
|      *            The list to put full paths in.
 | |
|      */
 | |
|     private void recursiveGetHeadPaths(AVMNode node, List<String> components, List<Pair<Integer, String>> paths)
 | |
|     {
 | |
|         if (!can(null, node, PermissionService.READ_CHILDREN, false))
 | |
|         {
 | |
|             return;
 | |
|         }
 | |
|         if (node.getIsRoot())
 | |
|         {
 | |
|             AVMStore store = fAVMStoreDAO.getByRoot(node);
 | |
|             if (store != null)
 | |
|             {
 | |
|                 addPath(components, -1, store.getName(), paths);
 | |
|                 return;
 | |
|             }
 | |
|             return;
 | |
|         }
 | |
|         List<ChildEntry> entries = fChildEntryDAO.getByChild(node);
 | |
|         for (ChildEntry entry : entries)
 | |
|         {
 | |
|             String name = entry.getKey().getName();
 | |
|             components.add(name);
 | |
|             AVMNode parent = entry.getKey().getParent();
 | |
|             recursiveGetHeadPaths(parent, components, paths);
 | |
|             components.remove(components.size() - 1);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Do the actual work.
 | |
|      * 
 | |
|      * @param node
 | |
|      *            The current node.
 | |
|      * @param components
 | |
|      *            The currently accumulated path components.
 | |
|      * @param paths
 | |
|      *            The list to put full paths in.
 | |
|      */
 | |
|     private void recursiveGetPathsInStoreHead(AVMNode node, List<String> components, List<Pair<Integer, String>> paths, DirectoryNode root, String storeName)
 | |
|     {
 | |
|         if (!can(null, node, PermissionService.READ_CHILDREN, false))
 | |
|         {
 | |
|             return;
 | |
|         }
 | |
|         if (node.equals(root))
 | |
|         {
 | |
|             addPath(components, -1, storeName, paths);
 | |
|             return;
 | |
|         }
 | |
|         List<ChildEntry> entries = fChildEntryDAO.getByChild(node);
 | |
|         for (ChildEntry entry : entries)
 | |
|         {
 | |
|             String name = entry.getKey().getName();
 | |
|             components.add(name);
 | |
|             AVMNode parent = entry.getKey().getParent();
 | |
|             recursiveGetHeadPaths(parent, components, paths);
 | |
|             components.remove(components.size() - 1);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Add a path to the list.
 | |
|      * 
 | |
|      * @param components
 | |
|      *            The path name components.
 | |
|      * @param version
 | |
|      *            The version id.
 | |
|      * @param storeName
 | |
|      *            The name of the
 | |
|      * @param paths
 | |
|      *            The List to add to.
 | |
|      */
 | |
|     private void addPath(List<String> components, int version, String storeName, List<Pair<Integer, String>> paths)
 | |
|     {
 | |
|         paths.add(new Pair<Integer, String>(version, makePath(components, storeName)));
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Alternate version.
 | |
|      * 
 | |
|      * @param components
 | |
|      * @param storeName
 | |
|      * @param paths
 | |
|      */
 | |
|     private void addPath(List<String> components, String storeName, List<String> paths)
 | |
|     {
 | |
|         paths.add(makePath(components, storeName));
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Helper for generating paths.
 | |
|      * 
 | |
|      * @param components
 | |
|      *            The path components.
 | |
|      * @param storeName
 | |
|      *            The store that the path is in.
 | |
|      * @return The path.
 | |
|      */
 | |
|     private String makePath(List<String> components, String storeName)
 | |
|     {
 | |
|         StringBuilder pathBuilder = new StringBuilder();
 | |
|         pathBuilder.append(storeName);
 | |
|         pathBuilder.append(":");
 | |
|         if (components.size() == 0)
 | |
|         {
 | |
|             pathBuilder.append("/");
 | |
|             return pathBuilder.toString();
 | |
|         }
 | |
|         for (int i = components.size() - 1; i >= 0; i--)
 | |
|         {
 | |
|             pathBuilder.append("/");
 | |
|             pathBuilder.append(components.get(i));
 | |
|         }
 | |
|         return pathBuilder.toString();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get information about layering of a path.
 | |
|      * 
 | |
|      * @param version
 | |
|      *            The version to look under.
 | |
|      * @param path
 | |
|      *            The full avm path.
 | |
|      * @return A LayeringDescriptor.
 | |
|      */
 | |
|     public LayeringDescriptor getLayeringInfo(int version, String path)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             Lookup lookup = store.lookup(version, pathParts[1], false, true);
 | |
|             if (lookup == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Path not found.");
 | |
|             }
 | |
|             if (!can(store, lookup.getCurrentNode(), PermissionService.READ_PROPERTIES, false))
 | |
|             {
 | |
|                 throw new AccessDeniedException("Not allowed to read properties: " + path);
 | |
|             }
 | |
|             return new LayeringDescriptor(!lookup.getDirectlyContained(), lookup.getAVMStore().getDescriptor(), lookup.getFinalStore().getDescriptor());
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Lookup a directory specifically.
 | |
|      * 
 | |
|      * @param version
 | |
|      *            The version to look under.
 | |
|      * @param path
 | |
|      *            The path to lookup.
 | |
|      * @return A lookup object.
 | |
|      */
 | |
|     public Lookup lookupDirectory(int version, String path)
 | |
|     {
 | |
|         Integer count = fLookupCount.get();
 | |
|         try
 | |
|         {
 | |
|             if (count == null)
 | |
|             {
 | |
|                 fLookupCount.set(1);
 | |
|             }
 | |
|             fLookupCount.set(fLookupCount.get() + 1);
 | |
|             if (fLookupCount.get() > 50)
 | |
|             {
 | |
|                 throw new AVMCycleException(MSG_CYCLE_ON_LOOKUP);
 | |
|             }
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 return null;
 | |
|             }
 | |
|             return store.lookupDirectory(version, pathParts[1], false);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             if (count == null)
 | |
|             {
 | |
|                 fLookupCount.set(null);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Utility to split a path, foo:bar/baz into its repository and path parts.
 | |
|      * 
 | |
|      * @param path
 | |
|      *            The fully qualified path.
 | |
|      * @return The repository name and the repository path.
 | |
|      */
 | |
|     private String[] SplitPath(String path)
 | |
|     {
 | |
|         return AVMUtil.splitPath(path);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Make a directory into a primary indirection.
 | |
|      * 
 | |
|      * @param path
 | |
|      *            The full path.
 | |
|      */
 | |
|     public void makePrimary(String path)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             store.makePrimary(pathParts[1]);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Change what a layered directory points at.
 | |
|      * 
 | |
|      * @param path
 | |
|      *            The full path to the layered directory.
 | |
|      * @param target
 | |
|      *            The new target path.
 | |
|      */
 | |
|     public void retargetLayeredDirectory(String path, String target)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             store.retargetLayeredDirectory(pathParts[1], target);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the history chain for a node.
 | |
|      * 
 | |
|      * @param desc
 | |
|      *            The node to get history of.
 | |
|      * @param count
 | |
|      *            The maximum number of ancestors to traverse. Negative means all.
 | |
|      * @return A List of ancestors.
 | |
|      */
 | |
|     public List<AVMNodeDescriptor> getHistory(AVMNodeDescriptor desc, int count)
 | |
|     {
 | |
|         AVMNode node = fAVMNodeDAO.getByID(desc.getId());
 | |
|         if (node == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Not found.");
 | |
|         }
 | |
|         if (!can(null, node, PermissionService.READ_PROPERTIES, false))
 | |
|         {
 | |
|             throw new AccessDeniedException("Not allowed to read properties: " + desc);
 | |
|         }
 | |
|         if (count < 0)
 | |
|         {
 | |
|             count = Integer.MAX_VALUE;
 | |
|         }
 | |
|         List<AVMNodeDescriptor> history = new ArrayList<AVMNodeDescriptor>();
 | |
|         for (int i = 0; i < count; i++)
 | |
|         {
 | |
|             AVMNode ancNode = node.getAncestor();
 | |
|             if (ancNode == null)
 | |
|             {
 | |
|                 break;
 | |
|             }
 | |
|             if (!can(null, ancNode, PermissionService.READ_PROPERTIES, false))
 | |
|             {
 | |
|                 break;
 | |
|             }
 | |
|             if ((node.getType() == AVMNodeType.LAYERED_FILE) && (ancNode.getType() == AVMNodeType.PLAIN_FILE))
 | |
|             {
 | |
|                 break;
 | |
|             }
 | |
|             history.add(ancNode.getDescriptor("UNKNOWN", "UNKNOWN", "UNKNOWN", -1));
 | |
|             node = ancNode;
 | |
|         }
 | |
|         return history;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Set the opacity of a layered directory. An opaque directory hides the things it points to via indirection.
 | |
|      * 
 | |
|      * @param path
 | |
|      *            The path to the layered directory.
 | |
|      * @param opacity
 | |
|      *            True is opaque; false is not.
 | |
|      */
 | |
|     public void setOpacity(String path, boolean opacity)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             store.setOpacity(pathParts[1], opacity);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Set a property on a node.
 | |
|      * 
 | |
|      * @param path
 | |
|      *            The path to the node.
 | |
|      * @param name
 | |
|      *            The name of the property.
 | |
|      * @param value
 | |
|      *            The value of the property.
 | |
|      */
 | |
|     public void setNodeProperty(String path, QName name, PropertyValue value)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             store.setNodeProperty(pathParts[1], name, value);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Set a collection of properties at once.
 | |
|      * 
 | |
|      * @param path
 | |
|      *            The path to the node.
 | |
|      * @param properties
 | |
|      *            The Map of QNames to PropertyValues.
 | |
|      */
 | |
|     public void setNodeProperties(String path, Map<QName, PropertyValue> properties)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             store.setNodeProperties(pathParts[1], properties);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get a property by name for a node.
 | |
|      * 
 | |
|      * @param version
 | |
|      *            The version to look under.
 | |
|      * @param path
 | |
|      *            The path to the node.
 | |
|      * @param name
 | |
|      *            The name of the property.
 | |
|      * @return The PropertyValue or null if it does not exist.
 | |
|      */
 | |
|     public PropertyValue getNodeProperty(int version, String path, QName name)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             return store.getNodeProperty(version, pathParts[1], name);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get a Map of all the properties of a node.
 | |
|      * 
 | |
|      * @param version
 | |
|      *            The version to look under.
 | |
|      * @param path
 | |
|      *            The path to the node.
 | |
|      * @return A Map of QNames to PropertyValues.
 | |
|      */
 | |
|     public Map<QName, PropertyValue> getNodeProperties(int version, String path)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             return store.getNodeProperties(version, pathParts[1]);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Delete a single property from a node.
 | |
|      * 
 | |
|      * @param path
 | |
|      *            The path to the node.
 | |
|      * @param name
 | |
|      *            The name of the property.
 | |
|      */
 | |
|     public void deleteNodeProperty(String path, QName name)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             store.deleteNodeProperty(pathParts[1], name);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Delete all properties on a node.
 | |
|      * 
 | |
|      * @param path
 | |
|      *            The path to the node.
 | |
|      */
 | |
|     public void deleteNodeProperties(String path)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             store.deleteNodeProperties(pathParts[1]);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Set a property on a store. Overwrites if property exists.
 | |
|      * 
 | |
|      * @param store
 | |
|      *            The AVMStore.
 | |
|      * @param name
 | |
|      *            The QName.
 | |
|      * @param value
 | |
|      *            The PropertyValue to set.
 | |
|      */
 | |
|     public void setStoreProperty(String store, QName name, PropertyValue value)
 | |
|     {
 | |
|         AVMStore st = getAVMStoreByName(store);
 | |
|         if (st == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Store not found.");
 | |
|         }
 | |
|         st.setProperty(name, value);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Set a group of properties on a store. Overwrites any properties that exist.
 | |
|      * 
 | |
|      * @param store
 | |
|      *            The AVMStore.
 | |
|      * @param props
 | |
|      *            The properties to set.
 | |
|      */
 | |
|     public void setStoreProperties(String store, Map<QName, PropertyValue> props)
 | |
|     {
 | |
|         AVMStore st = getAVMStoreByName(store);
 | |
|         if (st == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Store not found.");
 | |
|         }
 | |
|         st.setProperties(props);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get a property from a store.
 | |
|      * 
 | |
|      * @param store
 | |
|      *            The name of the store.
 | |
|      * @param name
 | |
|      *            The property
 | |
|      * @return The property value or null if non-existent.
 | |
|      */
 | |
|     public PropertyValue getStoreProperty(String store, QName name)
 | |
|     {
 | |
|         if (store == null)
 | |
|         {
 | |
|             throw new AVMBadArgumentException("Null store name.");
 | |
|         }
 | |
|         AVMStore st = getAVMStoreByName(store);
 | |
|         if (st == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Store not found.");
 | |
|         }
 | |
|         return st.getProperty(name);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Queries a given store for properties with keys that match a given pattern.
 | |
|      * 
 | |
|      * @param store
 | |
|      *            The name of the store.
 | |
|      * @param keyPattern
 | |
|      *            The sql 'like' pattern, inserted into a QName.
 | |
|      * @return A Map of the matching key value pairs.
 | |
|      */
 | |
|     public Map<QName, PropertyValue> queryStorePropertyKey(String store, QName keyPattern)
 | |
|     {
 | |
|         AVMStore st = getAVMStoreByName(store);
 | |
|         if (st == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Store not found.");
 | |
|         }
 | |
|         
 | |
|         return fAVMStorePropertyDAO.queryByKeyPattern(st, keyPattern);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Queries all AVM stores for properties with keys that match a given pattern.
 | |
|      * 
 | |
|      * @param keyPattern
 | |
|      *            The sql 'like' pattern, inserted into a QName.
 | |
|      * @return A List of Pairs of Store name, Map.Entry.
 | |
|      */
 | |
|     public Map<String, Map<QName, PropertyValue>> queryStoresPropertyKeys(QName keyPattern)
 | |
|     {
 | |
|         return fAVMStorePropertyDAO.queryByKeyPattern(keyPattern);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get all the properties for a store.
 | |
|      * 
 | |
|      * @param store
 | |
|      *            The name of the Store.
 | |
|      * @return A Map of all the properties.
 | |
|      */
 | |
|     public Map<QName, PropertyValue> getStoreProperties(String store)
 | |
|     {
 | |
|         if (store == null)
 | |
|         {
 | |
|             throw new AVMBadArgumentException("Null store name.");
 | |
|         }
 | |
|         AVMStore st = getAVMStoreByName(store);
 | |
|         if (st == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Store not found.");
 | |
|         }
 | |
|         return st.getProperties();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Delete a property from a store.
 | |
|      * 
 | |
|      * @param store
 | |
|      *            The name of the store.
 | |
|      * @param name
 | |
|      *            The name of the property.
 | |
|      */
 | |
|     public void deleteStoreProperty(String store, QName name)
 | |
|     {
 | |
|         AVMStore st = getAVMStoreByName(store);
 | |
|         if (st == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Store not found.");
 | |
|         }
 | |
|         st.deleteProperty(name);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the common ancestor of two nodes if one exists. Unfortunately this is a quadratic problem, taking time
 | |
|      * proportional to the product of the lengths of the left and right history chains.
 | |
|      * 
 | |
|      * @param left
 | |
|      *            The first node.
 | |
|      * @param right
 | |
|      *            The second node.
 | |
|      * @return The common ancestor. There are four possible results. Null means that there is no common ancestor. Left
 | |
|      *         returned means that left is strictly an ancestor of right. Right returned means that right is strictly an
 | |
|      *         ancestor of left. Any other non null return is the common ancestor and indicates that left and right are
 | |
|      *         in conflict.
 | |
|      */
 | |
|     public AVMNodeDescriptor getCommonAncestor(AVMNodeDescriptor left, AVMNodeDescriptor right)
 | |
|     {
 | |
|         AVMNode lNode = fAVMNodeDAO.getByID(left.getId());
 | |
|         AVMNode rNode = fAVMNodeDAO.getByID(right.getId());
 | |
|         if (lNode == null || rNode == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Node not found.");
 | |
|         }
 | |
|         if (!can(null, lNode, PermissionService.READ_PROPERTIES, false))
 | |
|         {
 | |
|             throw new AccessDeniedException("Not allowed to read properties: " + left);
 | |
|         }
 | |
|         if (!can(null, rNode, PermissionService.READ_PROPERTIES, false))
 | |
|         {
 | |
|             throw new AccessDeniedException("Not allowed to read properties: " + right);
 | |
|         }
 | |
|         // TODO Short changing the permissions checking here. I'm not sure
 | |
|         // if that's OK.
 | |
|         List<AVMNode> leftHistory = new ArrayList<AVMNode>();
 | |
|         List<AVMNode> rightHistory = new ArrayList<AVMNode>();
 | |
|         while (lNode != null || rNode != null)
 | |
|         {
 | |
|             boolean checkRight = false;
 | |
|             if (lNode != null)
 | |
|             {
 | |
|                 leftHistory.add(lNode);
 | |
|                 checkRight = true;
 | |
|                 lNode = lNode.getAncestor();
 | |
|             }
 | |
|             boolean checkLeft = false;
 | |
|             if (rNode != null)
 | |
|             {
 | |
|                 rightHistory.add(rNode);
 | |
|                 checkLeft = true;
 | |
|                 rNode = rNode.getAncestor();
 | |
|             }
 | |
|             if (checkRight)
 | |
|             {
 | |
|                 AVMNode check = leftHistory.get(leftHistory.size() - 1);
 | |
|                 for (AVMNode node : rightHistory)
 | |
|                 {
 | |
|                     if (node.equals(check))
 | |
|                     {
 | |
|                         return node.getDescriptor("", "", "", -1);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             if (checkLeft)
 | |
|             {
 | |
|                 AVMNode check = rightHistory.get(rightHistory.size() - 1);
 | |
|                 for (AVMNode node : leftHistory)
 | |
|                 {
 | |
|                     if (node.equals(check))
 | |
|                     {
 | |
|                         return node.getDescriptor("", "", "", -1);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         return null;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the ContentData for a file.
 | |
|      * 
 | |
|      * @param version
 | |
|      *            The version to look under.
 | |
|      * @param path
 | |
|      *            The path to the file.
 | |
|      * @return The ContentData for the file.
 | |
|      */
 | |
|     public ContentData getContentDataForRead(int version, String path)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             return store.getContentDataForRead(version, pathParts[1]);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the ContentData for a file for writing.
 | |
|      * 
 | |
|      * @param path
 | |
|      *            The path to the file.
 | |
|      * @return The ContentData object.
 | |
|      */
 | |
|     public ContentData getContentDataForWrite(String path)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             ContentData result = store.getContentDataForWrite(pathParts[1]);
 | |
|             return result;
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Set the ContentData on a file.
 | |
|      * 
 | |
|      * @param path
 | |
|      *            The path to the file.
 | |
|      * @param data
 | |
|      *            The content data to set.
 | |
|      */
 | |
|     public void setContentData(String path, ContentData data)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             store.setContentData(pathParts[1], data);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the single instance of AVMRepository.
 | |
|      * 
 | |
|      * @return The single instance.
 | |
|      */
 | |
|     public static AVMRepository GetInstance()
 | |
|     {
 | |
|         return fgInstance;
 | |
|     }
 | |
| 
 | |
|     public void setMetaDataFrom(String path, AVMNodeDescriptor from)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found: " + pathParts[0]);
 | |
|             }
 | |
|             AVMNode fromNode = fAVMNodeDAO.getByID(from.getId());
 | |
|             if (fromNode == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Node not found: " + from.getPath());
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             store.setMetaDataFrom(pathParts[1], fromNode);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Add an aspect to an AVM Node.
 | |
|      * 
 | |
|      * @param path
 | |
|      *            The path to the node.
 | |
|      * @param aspectName
 | |
|      *            The name of the aspect.
 | |
|      */
 | |
|     public void addAspect(String path, QName aspectName)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             store.addAspect(pathParts[1], aspectName);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get all the aspects on an AVM node.
 | |
|      * 
 | |
|      * @param version
 | |
|      *            The version to look under.
 | |
|      * @param path
 | |
|      *            The path to the node.
 | |
|      * @return A List of the QNames of the Aspects.
 | |
|      */
 | |
|     public Set<QName> getAspects(int version, String path)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             return store.getAspects(version, pathParts[1]);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Remove an aspect and all associated properties from a node.
 | |
|      * 
 | |
|      * @param path
 | |
|      *            The path to the node.
 | |
|      * @param aspectName
 | |
|      *            The name of the aspect.
 | |
|      */
 | |
|     public void removeAspect(String path, QName aspectName)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             store.removeAspect(pathParts[1], aspectName);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Does a node have a particular aspect.
 | |
|      * 
 | |
|      * @param version
 | |
|      *            The version to look under.
 | |
|      * @param path
 | |
|      *            The path to the node.
 | |
|      * @param aspectName
 | |
|      *            The name of the aspect.
 | |
|      * @return Whether the node has the aspect.
 | |
|      */
 | |
|     public boolean hasAspect(int version, String path, QName aspectName)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             return store.hasAspect(version, pathParts[1], aspectName);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Set the ACL on a node.
 | |
|      * 
 | |
|      * @param path
 | |
|      *            The path to the node.
 | |
|      * @param acl
 | |
|      *            The ACL to set.
 | |
|      */
 | |
|     public void setACL(String path, Acl acl)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             store.setACL(pathParts[1], acl);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the ACL on a node.
 | |
|      * 
 | |
|      * @param version
 | |
|      *            The version to look under.
 | |
|      * @param path
 | |
|      *            The path to the node.
 | |
|      * @return The ACL.
 | |
|      */
 | |
|     public Acl getACL(int version, String path)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             return store.getACL(version, pathParts[1]);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Link a node into a directory, directly.
 | |
|      * 
 | |
|      * @param parentPath
 | |
|      *            The path to the parent.
 | |
|      * @param name
 | |
|      *            The name to give the node.
 | |
|      * @param toLink
 | |
|      *            The node to link.
 | |
|      */
 | |
|     public void link(String parentPath, String name, AVMNodeDescriptor toLink)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(parentPath);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             store.link(pathParts[1], name, toLink);
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Update a link, directly.
 | |
|      * 
 | |
|      * @param parentPath
 | |
|      *            The path to the parent.
 | |
|      * @param name
 | |
|      *            The name to give the node.
 | |
|      * @param toLink
 | |
|      *            The node to link.
 | |
|      */
 | |
|     public void updateLink(String parentPath, String name, AVMNodeDescriptor toLink)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(parentPath);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             store.updateLink(pathParts[1], name, toLink);
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * This is the danger version of link. It must be called on a copied and unsnapshotted directory. It blithely
 | |
|      * inserts a child without checking if a child exists with a conflicting name.
 | |
|      * 
 | |
|      * @param parent
 | |
|      *            The parent directory.
 | |
|      * @param name
 | |
|      *            The name to give the child.
 | |
|      * @param child
 | |
|      *            The child to link in.
 | |
|      */
 | |
|     public void link(AVMNodeDescriptor parent, String name, AVMNodeDescriptor child)
 | |
|     {
 | |
|         AVMNode node = fAVMNodeDAO.getByID(parent.getId());
 | |
|         if (!(node instanceof DirectoryNode))
 | |
|         {
 | |
|             throw new AVMWrongTypeException("Not a Directory.");
 | |
|         }
 | |
|         DirectoryNode dir = (DirectoryNode) node;
 | |
|         if (!dir.getIsNew())
 | |
|         {
 | |
|             throw new AVMException("Directory has not already been copied.");
 | |
|         }
 | |
|         if (!can(null, dir, PermissionService.ADD_CHILDREN, false))
 | |
|         {
 | |
|             throw new AccessDeniedException("Not allowed to write: " + parent);
 | |
|         }
 | |
|         dir.link(name, child);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Remove name without leaving behind a deleted node. Dangerous if used unwisely.
 | |
|      * 
 | |
|      * @param path
 | |
|      *            The path to the layered directory.
 | |
|      * @param name
 | |
|      *            The name of the child.
 | |
|      */
 | |
|     public void flatten(String path, String name)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             fLookupCache.onDelete(pathParts[0]);
 | |
|             Lookup lPath = store.lookup(-1, pathParts[1], true, false);
 | |
|             AVMNode node = lPath.getCurrentNode();
 | |
|             if (node == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Path not found.");
 | |
|             }
 | |
|             if (!(node instanceof LayeredDirectoryNode))
 | |
|             {
 | |
|                 throw new AVMWrongTypeException("Not a Layered Directory.");
 | |
|             }
 | |
|             if (!can(store, node, PermissionService.FLATTEN, lPath.getDirectlyContained()))
 | |
|             {
 | |
|                 throw new AccessDeniedException("Not allowed to write in: " + path);
 | |
|             }
 | |
|             LayeredDirectoryNode dir = (LayeredDirectoryNode) node;
 | |
|             dir.flatten(name);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Force a copy on write.
 | |
|      * 
 | |
|      * @param path
 | |
|      *            The path to force.
 | |
|      */
 | |
|     public AVMNodeDescriptor forceCopy(String path)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found.");
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             // Just force a copy if needed by looking up in write mode.
 | |
|             Lookup lPath = store.lookup(-1, pathParts[1], true, true);
 | |
|             if (lPath == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Path not found.");
 | |
|             }
 | |
|             AVMNode node = lPath.getCurrentNode();
 | |
|             AVMNodeDescriptor desc = node.getDescriptor(lPath);
 | |
|             return desc;
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Rename a store.
 | |
|      * 
 | |
|      * @param sourceName
 | |
|      *            The original name.
 | |
|      * @param destName
 | |
|      *            The new name.
 | |
|      * @throws AVMNotFoundException
 | |
|      * @throws AVMExistsException
 | |
|      */
 | |
|     public void renameStore(String sourceName, String destName)
 | |
|     {
 | |
|         AlfrescoTransactionSupport.bindListener(fPurgeStoreTxnListener);
 | |
|         AlfrescoTransactionSupport.bindListener(fCreateStoreTxnListener);
 | |
|         AVMStore store = getAVMStoreByName(sourceName);
 | |
|         if (store == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Store Not Found: " + sourceName);
 | |
|         }
 | |
|         if (getAVMStoreByName(destName) != null)
 | |
|         {
 | |
|             throw new AVMExistsException("Store Already Exists: " + destName);
 | |
|         }
 | |
|         if (!FileNameValidator.isValid(destName))
 | |
|         {
 | |
|             throw new AVMBadArgumentException("Bad store name: " + destName);
 | |
|         }
 | |
|         store.setName(destName);
 | |
|         
 | |
|         AVMDAOs.Instance().fAVMStoreDAO.update(store);
 | |
|         
 | |
|         store.createSnapshot("Rename Store", "Rename Store from " + sourceName + " to " + destName, new HashMap<String, Integer>());
 | |
|         fLookupCache.onDelete(sourceName);
 | |
|         fAVMStoreDAO.invalidateCache();
 | |
|         fPurgeStoreTxnListener.storePurged(sourceName);
 | |
|         fCreateStoreTxnListener.storeCreated(destName);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Revert a head path to a given version. This works by cloning the version to revert to, and then linking that new
 | |
|      * version into head. The reverted version will have the previous head version as ancestor.
 | |
|      * 
 | |
|      * @param path
 | |
|      *            The path to the parent directory.
 | |
|      * @param name
 | |
|      *            The name of the node.
 | |
|      * @param toRevertTo
 | |
|      *            The descriptor of the version to revert to.
 | |
|      */
 | |
|     public void revert(String path, String name, AVMNodeDescriptor toRevertTo)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found: " + pathParts[0]);
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             store.revert(pathParts[1], name, toRevertTo);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Set the GUID on a node.
 | |
|      * 
 | |
|      * @param path
 | |
|      * @param guid
 | |
|      */
 | |
|     public void setGuid(String path, String guid)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store not found:" + pathParts[0]);
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             store.setGuid(pathParts[1], guid);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Set the encoding on a node.
 | |
|      * 
 | |
|      * @param path
 | |
|      * @param encoding
 | |
|      */
 | |
|     public void setEncoding(String path, String encoding)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store Not Found: " + pathParts[0]);
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             store.setEncoding(pathParts[1], encoding);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Set the mime type on a node.
 | |
|      * 
 | |
|      * @param path
 | |
|      * @param encoding
 | |
|      */
 | |
|     public void setMimeType(String path, String mimeType)
 | |
|     {
 | |
|         fLookupCount.set(1);
 | |
|         try
 | |
|         {
 | |
|             String[] pathParts = SplitPath(path);
 | |
|             AVMStore store = getAVMStoreByName(pathParts[0]);
 | |
|             if (store == null)
 | |
|             {
 | |
|                 throw new AVMNotFoundException("Store Not Found: " + pathParts[0]);
 | |
|             }
 | |
|             fLookupCache.onWrite(pathParts[0]);
 | |
|             store.setMimeType(pathParts[1], mimeType);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             fLookupCount.set(null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public List<String> getPathsInStoreVersion(AVMNodeDescriptor desc, String store, int version)
 | |
|     {
 | |
|         AVMNode node = fAVMNodeDAO.getByID(desc.getId());
 | |
|         if (node == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Not found: " + desc);
 | |
|         }
 | |
|         List<String> paths = new ArrayList<String>();
 | |
|         List<String> components = new ArrayList<String>();
 | |
|         recursiveGetStoreVersionPaths(store, node, version, components, paths);
 | |
|         return paths;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Do the actual work.
 | |
|      * 
 | |
|      * @param node
 | |
|      *            The current node.
 | |
|      * @param components
 | |
|      *            The currently accumulated path components.
 | |
|      * @param paths
 | |
|      *            The list to put full paths in.
 | |
|      */
 | |
|     private void recursiveGetStoreVersionPaths(String storeName, AVMNode node, int version, List<String> components, List<String> paths)
 | |
|     {
 | |
|         if (!can(null, node, PermissionService.READ, false))
 | |
|         {
 | |
|             return;
 | |
|         }
 | |
|         if (node.getIsRoot())
 | |
|         {
 | |
|             VersionRoot versionRoot = fVersionRootDAO.getByRoot(node);
 | |
|             if (versionRoot.getAvmStore().getName().equals(storeName) && versionRoot.getVersionID() == version)
 | |
|             {
 | |
|                 addPath(components, storeName, paths);
 | |
|                 return;
 | |
|             }
 | |
|             return;
 | |
|         }
 | |
|         List<ChildEntry> entries = fChildEntryDAO.getByChild(node);
 | |
|         for (ChildEntry entry : entries)
 | |
|         {
 | |
|             String name = entry.getKey().getName();
 | |
|             components.add(name);
 | |
|             AVMNode parent = entry.getKey().getParent();
 | |
|             recursiveGetStoreVersionPaths(storeName, parent, version, components, paths);
 | |
|             components.remove(components.size() - 1);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public Map<QName, PropertyValue> getNodeProperties(AVMNodeDescriptor desc)
 | |
|     {
 | |
|         AVMNode node = fAVMNodeDAO.getByID(desc.getId());
 | |
|         if (node == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Node not found: " + desc);
 | |
|         }
 | |
|         if (!can(null, node, PermissionService.READ_PROPERTIES, false))
 | |
|         {
 | |
|             throw new AccessDeniedException("Not allowed to read properties: " + desc);
 | |
|         }
 | |
|         return node.getProperties();
 | |
|     }
 | |
| 
 | |
|     public ContentData getContentDataForRead(AVMNodeDescriptor desc)
 | |
|     {
 | |
|         AVMNode node = fAVMNodeDAO.getByID(desc.getId());
 | |
|         if (node == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Node not found: " + desc);
 | |
|         }
 | |
|         if (!can(null, node, PermissionService.READ_CONTENT, false))
 | |
|         {
 | |
|             throw new AccessDeniedException("Not allowed to read: " + desc);
 | |
|         }
 | |
|         if (node.getType() == AVMNodeType.PLAIN_FILE)
 | |
|         {
 | |
|             PlainFileNode file = (PlainFileNode) node;
 | |
|             return file.getContentData();
 | |
|         }
 | |
|         else if (node.getType() == AVMNodeType.LAYERED_FILE)
 | |
|         {
 | |
|             LayeredFileNode file = (LayeredFileNode) node;
 | |
|             return file.getContentData(null);
 | |
|         }
 | |
|         throw new AVMWrongTypeException("Not a file: " + desc);
 | |
|     }
 | |
| 
 | |
|     public Set<QName> getAspects(AVMNodeDescriptor desc)
 | |
|     {
 | |
|         AVMNode node = fAVMNodeDAO.getByID(desc.getId());
 | |
|         if (node == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Node not found: " + desc);
 | |
|         }
 | |
|         if (!can(null, node, PermissionService.READ_PROPERTIES, false))
 | |
|         {
 | |
|             throw new AccessDeniedException("Not allowed to read properties: " + desc);
 | |
|         }
 | |
|         Set<QName> aspectQNames = node.getAspects();
 | |
|         return aspectQNames;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Evaluate permission on a node. I've got a bad feeling about this...
 | |
|      * 
 | |
|      * @param store
 | |
|      * @param node
 | |
|      * @param permission
 | |
|      * @return
 | |
|      */
 | |
|     public boolean can(AVMStore store, AVMNode node, String permission, boolean isDirectlyContained)
 | |
|     {
 | |
|         Acl acl = node.getAcl();
 | |
| 
 | |
|         QName type;
 | |
|         if (node.getType() == AVMNodeType.PLAIN_DIRECTORY)
 | |
|         {
 | |
|             type = WCMModel.TYPE_AVM_PLAIN_FOLDER;
 | |
|         }
 | |
|         else if (node.getType() == AVMNodeType.PLAIN_FILE)
 | |
|         {
 | |
|             type = WCMModel.TYPE_AVM_PLAIN_CONTENT;
 | |
|         }
 | |
|         else if (node.getType() == AVMNodeType.LAYERED_DIRECTORY)
 | |
|         {
 | |
|             type = WCMModel.TYPE_AVM_LAYERED_FOLDER;
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             type = WCMModel.TYPE_AVM_LAYERED_CONTENT;
 | |
|         }
 | |
|         PermissionContext context = new PermissionContext(type);
 | |
|         
 | |
|         context.addDynamicAuthorityAssignment(node.getBasicAttributes().getOwner(), PermissionService.OWNER_AUTHORITY);
 | |
| 
 | |
|         if ((store != null) && isDirectlyContained)
 | |
|         {
 | |
|             String storeOwner = getStoreUserName(store.getName());
 | |
|             if (storeOwner != null)
 | |
|             {
 | |
|                 // Special case: WCM author sandbox (author, author preview, author workflow, author workflow preview)
 | |
|                 context.addDynamicAuthorityAssignment(storeOwner, PermissionService.WCM_STORE_OWNER_AUTHORITY);
 | |
|             }
 | |
|             
 | |
|             /*
 | |
|             StoreType storeType = StoreType.getStoreType(store.getName(), store.getDescriptor(), store.getProperties());
 | |
|             switch (storeType)
 | |
|             {
 | |
|             case AUTHOR:
 | |
|             case AUTHOR_PREVIEW:
 | |
|             case AUTHOR_WORKFLOW:
 | |
|             case AUTHOR_WORKFLOW_PREVIEW:
 | |
|                 String storeName = store.getName();
 | |
|                 int first = -1;
 | |
|                 int second = -1;
 | |
|                 first = storeName.indexOf("--");
 | |
|                 if (first >= 0)
 | |
|                 {
 | |
|                     second = storeName.indexOf("--", first + 2);
 | |
|                     String storeOwner;
 | |
|                     if (second >= 0)
 | |
|                     {
 | |
|                          storeOwner = storeName.substring(first + 2, second);
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         storeOwner = storeName.substring(first + 2);
 | |
|                     }
 | |
|                     context.addDynamicAuthorityAssignment(storeOwner, PermissionService.WCM_STORE_OWNER_AUTHORITY);
 | |
|                 }
 | |
|                 break;
 | |
|             case STAGING:
 | |
|             case STAGING_PREVIEW:
 | |
|             case UNKNOWN:
 | |
|             case WORKFLOW:
 | |
|             case WORKFLOW_PREVIEW:
 | |
|             default:
 | |
|             }
 | |
|             */
 | |
|         }
 | |
|         
 | |
|         // Pass in node aspects
 | |
|         Set<QName> nodeAspectQNames = node.getAspects();
 | |
|         Set<QName> contextQNames = context.getAspects();
 | |
|         contextQNames.addAll(nodeAspectQNames);
 | |
|         
 | |
|         /* TODO review - PermissionContext.getProperties() not currently used ?
 | |
|         // Pass in node properties
 | |
|         Map<Long, PropertyValue> nodeProperties = node.getProperties();
 | |
|         Map<QName, Serializable> contextProperties = new HashMap<QName, Serializable>(5);
 | |
|         QNameDAO qnameDAO = AVMDAOs.Instance().fQNameDAO;
 | |
|         for (Map.Entry<Long, PropertyValue> entry : nodeProperties.entrySet())
 | |
|         {
 | |
|             QName qname = qnameDAO.getQName(entry.getKey()).getSecond();
 | |
|             PropertyDefinition def = fDictionaryService.getProperty(qname);
 | |
|             if (def == null)
 | |
|             {
 | |
|                 contextProperties.put(qname, entry.getValue().getValue(DataTypeDefinition.ANY));
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 contextProperties.put(qname, entry.getValue().getValue(def.getDataType().getName()));
 | |
|             }
 | |
|         }
 | |
|         context.getProperties().putAll(contextProperties);
 | |
|         */
 | |
|         
 | |
|         Long aclId = null;
 | |
|         if (acl != null)
 | |
|         {
 | |
|             aclId = acl.getId();
 | |
|         }
 | |
|         if (store != null)
 | |
|         {
 | |
|             Acl storeAcl = store.getStoreAcl();
 | |
|             if (storeAcl != null)
 | |
|             {
 | |
|                 Long storeAclID = storeAcl.getId();
 | |
|                 context.setStoreAcl(storeAclID);
 | |
|             }
 | |
|         }
 | |
|         return fPermissionService.hasPermission(aclId, context, permission) == AccessStatus.ALLOWED;
 | |
|     }
 | |
|     
 | |
|     private final static String WCM_STORE_SEPARATOR = "--";
 | |
|     private final static String WCM_STORE_PREVIEW   = "--preview";
 | |
|     private final static String WCM_STORE_WORKFLOW  = "--workflow-";
 | |
|    
 | |
|     // TODO - merge with WCM 3.x utils
 | |
|     // 
 | |
|     // Note: relies on WCM sandbox naming convention
 | |
|     //
 | |
|     // Staging:                 mystore
 | |
|     // Staging preview:         mystore--preview
 | |
|     // Author:                  mystore--myuserid
 | |
|     // Author preview:          mystore--myuserid--preview
 | |
|     // Author workflow:         mystore--myuserid--workflow-guid
 | |
|     // Author workflow preview: mystore--myuserid--workflow-guid--preview
 | |
|     // Workflow:                mystore--workflow-guid
 | |
|     // Workflow preview:        mystore--worklow-guid--preview
 | |
|     //
 | |
|     private static String getStoreUserName(String storeName)
 | |
|     {
 | |
|        String storeOwner = null;
 | |
|        int preview = storeName.indexOf(WCM_STORE_PREVIEW);
 | |
|        if (preview >= 0)
 | |
|        {
 | |
|            // strip off "--preview"
 | |
|            storeName = storeName.substring(0, preview);
 | |
|        }
 | |
|        int workflow = storeName.indexOf(WCM_STORE_WORKFLOW);
 | |
|        if (workflow >= 0)
 | |
|        {
 | |
|            // strip off "--workflow-<guid>"
 | |
|            storeName = storeName.substring(0, workflow);
 | |
|        }
 | |
|        int author = storeName.indexOf(WCM_STORE_SEPARATOR);
 | |
|        if (author >= 0)
 | |
|        {
 | |
|            storeOwner = storeName.substring(author + 2);
 | |
|        }
 | |
|        return storeOwner;
 | |
|     }
 | |
|     
 | |
|     public boolean can(String storeName, int version, String path, String permission)
 | |
|     {
 | |
|         Lookup lookup = AVMRepository.GetInstance().lookup(version, path, true);
 | |
|         if (lookup != null)
 | |
|         {
 | |
|             AVMNode node = lookup.getCurrentNode();
 | |
|             AVMStore store = getAVMStoreByName(storeName);
 | |
|             return can(store, node, permission, lookup.getDirectlyContained());
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             // Does not exist => allowed
 | |
|             return true;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Set the acl on a store.
 | |
|      * 
 | |
|      * @param storeName
 | |
|      * @param acl
 | |
|      */
 | |
|     public void setStoreAcl(String storeName, Acl acl)
 | |
|     {
 | |
|         AVMStore store = getAVMStoreByName(storeName);
 | |
|         if (store == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Store not found: " + storeName);
 | |
|         }
 | |
|         store.setStoreAcl(acl);
 | |
|         
 | |
|         AVMDAOs.Instance().fAVMStoreDAO.update(store);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the ACL on a store.
 | |
|      * 
 | |
|      * @param storeName
 | |
|      * @return
 | |
|      */
 | |
|     public Acl getStoreAcl(String storeName)
 | |
|     {
 | |
|         AVMStore store = getAVMStoreByName(storeName);
 | |
|         if (store == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Store not found: " + storeName);
 | |
|         }
 | |
|         return store.getStoreAcl();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param name
 | |
|      * @param version
 | |
|      * @return
 | |
|      */
 | |
|     public List<VersionDescriptor> getAVMStoreVersionsTo(String name, int version)
 | |
|     {
 | |
|         AVMStore store = getAVMStoreByName(name);
 | |
|         if (store == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Store not found.");
 | |
|         }
 | |
|         return store.getVersionsTo(version);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param name
 | |
|      * @param version
 | |
|      * @return
 | |
|      */
 | |
|     public List<VersionDescriptor> getAVMStoreVersionsFrom(String name, int version)
 | |
|     {
 | |
|         AVMStore store = getAVMStoreByName(name);
 | |
|         if (store == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Store not found.");
 | |
|         }
 | |
|         return store.getVersionsFrom(version);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param name
 | |
|      * @param startVersion
 | |
|      * @param endVersion
 | |
|      * @return
 | |
|      */
 | |
|     public List<VersionDescriptor> getAVMStoreVersionsBetween(String name, int startVersion, int endVersion)
 | |
|     {
 | |
|         AVMStore store = getAVMStoreByName(name);
 | |
|         if (store == null)
 | |
|         {
 | |
|             throw new AVMNotFoundException("Store not found.");
 | |
|         }
 | |
|         return store.getVersionsBetween(startVersion, endVersion);
 | |
|     }
 | |
| }
 |