mirror of
				https://github.com/Alfresco/alfresco-community-repo.git
				synced 2025-10-22 15:12:38 +00:00 
			
		
		
		
	29870: ALF-9952: Compatibility fix to allow mounting of Alfresco WebDAV as a network drive on Windows XP (without WebDrive)
   29872: Fixed ALF-7698 "Defects in tags picker in SHARE."
   - now checks added items to avoid duplicates
   29873: Made sure the onActionDetails doclib action generates SimpleDialog instance ids each time, to avoid multiple js class instances with the same id getting called for every dialog created.
   29878: ALF-9378: Use caching plus a read write lock, in a similar manner to ALF-7064, to avoid contention in SubsystemProxyFactory under load
   29881: Added helper text for the translators to the property file.
   29910: ALF-7433: a file deleted using the web UI still appears in a NFS mount but with NULL stats
   - timestamp propagation now enabled by default so NFS always works
   - timestamp propagation less agressive - only has to propagate on adds / deletes and renames
   29911: Values for recurrence strings updated now that I've read RFC-2445
   29944: ALF-9988: Merged PATCHES/V3.3.4 to V3.4-BUG-FIX
      28581: ALF-8944: Corrected sort parameter handling for datetime fields, based on analyzer and tokenization mode
   29955: Fixed ALF-9965: Property names matching fails if the property contains non Alphanumeric characters
   29956: Fixed ALF-9424: Webform(XSD): xf:switch switch group does not work based on "complexContent with Base" element.
   29964: Fixed ALF-9089: TinyMCE convert_fonts_to_spans parameter is not being picked up in Web Forms
   29977: L10N Updates from Gloria (based on rev29838).
   29980: Ignore virtual tomcat directory
   29981: Merged BRANCHES/DEV/BELARUS/V3.4-BUG-FIX-2011_08_19 to BRANCHES/DEV/V3.4-BUG-FIX 
       ALF-6808 : Incorrect Search Language Conversion for wild card characters (%, _) for MS SQL 
   29984: ALF-4753: Fix search query escaping in Alfresco Explorer
   - AWC-1743 / CHK-2171 fix restored
   29993: Merged DEV/TEMPORARY to V3.4-BUG-FIX
      29989: ALF-9976: WebDAV Class 2 compliance issue with OPTIONS response.
         Class 2 incompliance connected with absence of 'PROPPATCH' method in the 'Allow' response header has been fixed
   30009: ALF-7239 : Documents from Records Manager cannot be viewed in Outlook   
   30012: Merged BRANCHES/DEV/BELARUS/V3.4-BUG-FIX-2011_08_19 to BRANCHES/DEV/V3.4-BUG-FIX:
      29986: ALF-7105: pdfbox returns errors in the logs but one cannot understand what file is affected (PDFBox)
   30014: ALF-6284: Fix for Share Kerberos SSO Websphere compatibility (by Pavel)
   30016: Merged DEV to V3.4-BUG-FIX
      29424: ALF-8715: NFS: Admin sometimes cannot edit content uploaded via JSF
             Fix for "Can't open file for writing" during saving file in VI-like editors connected with FileExpiry daemon:
             - 'getNodeOrNull()' method extracted to get and check on existence of the cached file object
                and then update cache timeout or remove it from the cache;
             - 'getNodeForPath()' was appropriately updated in accordance with logic extracted to 'getNodeOrNull()';
             - 'fileExists()' method was corrected to take into account existence of the cached file object 
   30017: Reverses an accidental check in made with the last L10N bundle update (r29977)
   30045: ALF-8664 - Custom 'mandatory' RM metadata causes editing issues   
   30053: ALF-9681: webScriptsRegistryCache size in ehcache-default.xml   
      webScriptsRegistryCache maxElementsInMemory size was incremented up to 1000
   30079: ALF-10027: Fix template alfresco-global.properties for custom tomcat ports
   - Obselete web.application.context.url removed
   - alfresco.host, alfresco.port, share.host and share.port added
   30080: Fixed parse error mentioned in ALF-9511 "RSS Feeds with HTTP Auth doesn't work with Feed dashlet"
    - In java regexps the . (dot) does NOT include the \r or \n by default, to change that a (?s) instruction was added in the beginning of the regexp
   30083: Fixed ALF-10048 "Multiple thumbnails in RSS feeds causes problem for RSS Feed dashlet"
   30088: ALF-7433: Fix unit test fallout from new timestamp propagation behaviour
   30089: ALF-7433: Further unit test fallout from new timestamp propagation behaviour
   - testPermissionsAndPolicies needs to get folder modification date in a distinct transaction after populating it
   30091: ALF-10050: CIFS: Coordinator is unable to delete content after IMAP has been enabled
   - Transactional cache null entry issue
   30096: ALF-9793: Allow auto-created NFS authenticated users to work
   - Also removed admin from default mappings to avoid security hole
   30102: ALF-8723 CIFS on Windows fails to start under certain conditions
      - Found another place >= was required rather than just >
   30104: ALF-9890 SiteService is not producing audit data.
      - Internal siteService rather than public SiteService was being used, so there was no audit advice.
   30119: ALF-9793: Fallout from previous fix. Prevent initialization error when there are no user mappings and NFS is not enabled.
   30120: ALF-9526: Work around the fact that Websphere ignores the response encoding unless you call response.setCharacterEncoding()
   - GlobalLocalizationFilter inserts a response wrapper that parses the charset parameter from the content type header, if it is present
   30121: ALF-9535: Alfresco ignores Accept-Language sent from Share
   - Ticket and webscript authentication (i.e. Share requests) will use the language set in request headers as the session language
   30130: ALF-10049: Fix by Andrey to enable Flash upload on Weblogic!
   30141: ALF-8732 Now if the adhoc workflow throws exceptions while sending a notification email then those exceptions are ignored.
   30226: ALF-9415: JSF - Copy/Paste vs Cut/Paste add aspect rule.
   30242: Added NodeService.getNodeRef(Long nodeId)
    - First step to remove Lucene search from inbound email server
    - Done while investigating ALF-9660
   30243: Fixed ALF-9660: Inbound emails are always stored with encoding=UTF-8 regardless the encoding the email client uses.
    - If the encoding is provided, then it is not guessed
    - Includes tweak to remove Lucene search and rather use direct node ID-NodeRef translation provided by NodeService
   30263: Incremented version revision
   30264: ALF-10187: Merged V3.3 to V3.4-BUG-FIX
      30003: ALF-9898: More defensive exception handling to avoid packet pool leaks and extra logging on packet pool exhaustion
   30265: Merged V3.4 to V3.4-BUG-FIX
      30259: ALF-6527 LangPack FR - [Search results' page] Incorrect label when search matches results
        - Added "(s)" to "trouvé(s) dans l'entrepôt."   (for a repository search - found in retest)
        - The original change:
          Added "(s)" to "trouvé(s) dans tous les sites."   (for an All Sites search)
          
          However this leaves the current site search. As a result I also:
          Added "(s)" to "trouvé(s) dans le site {0}."  (for a current Site search)
          and checked with a French speaker that this would sound correct.
          
          Note: all three labels are prefixed by a separate text label: "N résultat(s) "
      30261: (RECORD ONLY) Merged V3.4-BUG-FIX to V3.4 (3.4.5)
      30262: ALF-10186: Merged PATCHES/V3.4.1 to V3.4
         30126: ALF-10075: Concurrency during CMIS document update causes content stream to close prematurely.
            - Created BufferedRequest request wrapper in RepositoryContainer
            - If the content stream is accessed directly it is streamed to a temporary file so that requests can be retried
         30137: ALF-10075: Fixed NPE
git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@30271 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
		
	
		
			
				
	
	
		
			1168 lines
		
	
	
		
			50 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
			
		
		
	
	
			1168 lines
		
	
	
		
			50 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
| /*
 | |
|  * Copyright (C) 2005-2011 Alfresco Software Limited.
 | |
|  *
 | |
|  * This file is part of Alfresco
 | |
|  *
 | |
|  * Alfresco is free software: you can redistribute it and/or modify
 | |
|  * it under the terms of the GNU Lesser General Public License as published by
 | |
|  * the Free Software Foundation, either version 3 of the License, or
 | |
|  * (at your option) any later version.
 | |
|  *
 | |
|  * Alfresco is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  * GNU Lesser General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU Lesser General Public License
 | |
|  * along with Alfresco. If not, see <http://www.gnu.org/licenses/>.
 | |
|  */
 | |
| package org.alfresco.repo.model.filefolder;
 | |
| 
 | |
| import java.io.InputStream;
 | |
| import java.io.InputStreamReader;
 | |
| import java.io.OutputStream;
 | |
| import java.io.Reader;
 | |
| import java.util.ArrayList;
 | |
| import java.util.Date;
 | |
| import java.util.List;
 | |
| import java.util.Locale;
 | |
| 
 | |
| import javax.transaction.Status;
 | |
| import javax.transaction.UserTransaction;
 | |
| 
 | |
| import junit.framework.TestCase;
 | |
| 
 | |
| import org.alfresco.error.AlfrescoRuntimeException;
 | |
| import org.alfresco.model.ContentModel;
 | |
| import org.alfresco.model.ForumModel;
 | |
| import org.alfresco.query.PagingRequest;
 | |
| import org.alfresco.query.PagingResults;
 | |
| import org.alfresco.repo.content.MimetypeMap;
 | |
| import org.alfresco.repo.dictionary.DictionaryDAO;
 | |
| import org.alfresco.repo.dictionary.M2Model;
 | |
| import org.alfresco.repo.dictionary.M2Type;
 | |
| import org.alfresco.repo.domain.node.AbstractNodeDAOImpl;
 | |
| import org.alfresco.repo.node.integrity.IntegrityChecker;
 | |
| import org.alfresco.repo.security.authentication.AuthenticationUtil;
 | |
| import org.alfresco.repo.security.authentication.AuthenticationUtil.RunAsWork;
 | |
| import org.alfresco.repo.security.permissions.AccessDeniedException;
 | |
| import org.alfresco.service.ServiceRegistry;
 | |
| import org.alfresco.service.cmr.model.FileExistsException;
 | |
| import org.alfresco.service.cmr.model.FileFolderService;
 | |
| import org.alfresco.service.cmr.model.FileFolderServiceType;
 | |
| import org.alfresco.service.cmr.model.FileInfo;
 | |
| import org.alfresco.service.cmr.model.FileNotFoundException;
 | |
| import org.alfresco.service.cmr.repository.ChildAssociationRef;
 | |
| import org.alfresco.service.cmr.repository.ContentReader;
 | |
| import org.alfresco.service.cmr.repository.ContentWriter;
 | |
| import org.alfresco.service.cmr.repository.CyclicChildRelationshipException;
 | |
| import org.alfresco.service.cmr.repository.NodeRef;
 | |
| import org.alfresco.service.cmr.repository.NodeService;
 | |
| import org.alfresco.service.cmr.repository.StoreRef;
 | |
| import org.alfresco.service.cmr.security.MutableAuthenticationService;
 | |
| import org.alfresco.service.cmr.security.PermissionService;
 | |
| import org.alfresco.service.cmr.view.ImporterService;
 | |
| import org.alfresco.service.cmr.view.Location;
 | |
| import org.alfresco.service.namespace.NamespaceService;
 | |
| import org.alfresco.service.namespace.QName;
 | |
| import org.alfresco.service.namespace.RegexQNamePattern;
 | |
| import org.alfresco.service.transaction.TransactionService;
 | |
| import org.alfresco.util.ApplicationContextHelper;
 | |
| import org.alfresco.util.GUID;
 | |
| import org.springframework.context.ApplicationContext;
 | |
| import org.springframework.extensions.surf.util.I18NUtil;
 | |
| 
 | |
| /**
 | |
|  * @see org.alfresco.repo.model.filefolder.FileFolderServiceImpl
 | |
|  * @author Derek Hulley
 | |
|  */
 | |
| public class FileFolderServiceImplTest extends TestCase
 | |
| {
 | |
|     private static final String IMPORT_VIEW = "filefolder/filefolder-test-import.xml";
 | |
| 
 | |
|     private static final String NAME_L0_FILE_A = "L0- File A";
 | |
|     private static final String NAME_L0_FILE_B = "L0- File B";
 | |
|     private static final String NAME_L0_FOLDER_A = "L0- Folder A";
 | |
|     private static final String NAME_L0_FOLDER_B = "L0- Folder B";
 | |
|     private static final String NAME_L0_FOLDER_C = "L0- Folder C";
 | |
|     private static final String NAME_L1_FOLDER_A = "L1- Folder A";
 | |
|     private static final String NAME_L1_FOLDER_B = "L1- Folder B";
 | |
|     private static final String NAME_L1_FILE_A = "L1- File A";
 | |
|     private static final String NAME_L1_FILE_B = "L1- File B";
 | |
|     private static final String NAME_L1_FILE_C = "L1- File C (%_)";
 | |
|     private static final String NAME_CHECK_FILE = "CHECK_FILE";
 | |
|     private static final String NAME_CHECK_FOLDER = "CHECK_FOLDER";
 | |
|     private static final String NAME_DISCUSSION_FOLDER = "CHECK_DISCUSSION_RENAME";
 | |
| 
 | |
|     private static final ApplicationContext ctx = ApplicationContextHelper.getApplicationContext();
 | |
| 
 | |
|     private TransactionService transactionService;
 | |
|     private NodeService nodeService;
 | |
|     private FileFolderService fileFolderService;
 | |
|     private PermissionService permissionService;
 | |
|     private MutableAuthenticationService authenticationService;
 | |
|     private DictionaryDAO dictionaryDAO;
 | |
|     private UserTransaction txn;
 | |
|     private NodeRef rootNodeRef;
 | |
|     private NodeRef workingRootNodeRef;
 | |
| 
 | |
|     @Override
 | |
|     public void setUp() throws Exception
 | |
|     {
 | |
|         ServiceRegistry serviceRegistry = (ServiceRegistry) ctx.getBean("ServiceRegistry");
 | |
|         transactionService = serviceRegistry.getTransactionService();
 | |
|         nodeService = serviceRegistry.getNodeService();
 | |
|         fileFolderService = serviceRegistry.getFileFolderService();
 | |
|         permissionService = serviceRegistry.getPermissionService();
 | |
|         authenticationService = (MutableAuthenticationService) ctx.getBean("AuthenticationService");
 | |
|         dictionaryDAO = (DictionaryDAO) ctx.getBean("dictionaryDAO");
 | |
| 
 | |
|         // start the transaction
 | |
|         txn = transactionService.getUserTransaction();
 | |
|         txn.begin();
 | |
| 
 | |
|         // downgrade integrity
 | |
|         IntegrityChecker.setWarnInTransaction();
 | |
| 
 | |
|         // authenticate
 | |
|         AuthenticationUtil.setFullyAuthenticatedUser(AuthenticationUtil.getAdminUserName());
 | |
| 
 | |
|         // create a test store
 | |
|         StoreRef storeRef = nodeService
 | |
|                 .createStore(StoreRef.PROTOCOL_WORKSPACE, getName() + System.currentTimeMillis());
 | |
|         rootNodeRef = nodeService.getRootNode(storeRef);
 | |
| 
 | |
|         // create a folder to import into
 | |
|         workingRootNodeRef = nodeService.createNode(
 | |
|                 rootNodeRef,
 | |
|                 ContentModel.ASSOC_CHILDREN,
 | |
|                 QName.createQName(NamespaceService.ALFRESCO_URI, "working root"),
 | |
|                 ContentModel.TYPE_FOLDER).getChildRef();
 | |
| 
 | |
|         // import the test data
 | |
|         ImporterService importerService = serviceRegistry.getImporterService();
 | |
|         Location importLocation = new Location(workingRootNodeRef);
 | |
|         InputStream is = getClass().getClassLoader().getResourceAsStream(IMPORT_VIEW);
 | |
|         if (is == null)
 | |
|         {
 | |
|             throw new NullPointerException("Test resource not found: " + IMPORT_VIEW);
 | |
|         }
 | |
|         Reader reader = new InputStreamReader(is);
 | |
|         importerService.importView(reader, importLocation, null, null);
 | |
|     }
 | |
| 
 | |
|     public void tearDown() throws Exception
 | |
|     {
 | |
|         try
 | |
|         {
 | |
|             if (txn.getStatus() != Status.STATUS_ROLLEDBACK && txn.getStatus() != Status.STATUS_COMMITTED)
 | |
|             {
 | |
|                 txn.rollback();
 | |
|             }
 | |
|         }
 | |
|         catch (Throwable e)
 | |
|         {
 | |
|             e.printStackTrace();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Checks that the names and numbers of files and folders in the provided list is correct
 | |
|      * 
 | |
|      * @param files the list of files
 | |
|      * @param expectedFileCount the number of uniquely named files expected
 | |
|      * @param expectedFolderCount the number of uniquely named folders expected
 | |
|      * @param expectedNames the names of the files and folders expected
 | |
|      */
 | |
|     private void checkFileList(
 | |
|             List<FileInfo> files,
 | |
|             int expectedFileCount,
 | |
|             int expectedFolderCount,
 | |
|             String[] expectedNames)
 | |
|     {
 | |
|         int fileCount = 0;
 | |
|         int folderCount = 0;
 | |
|         List<String> check = new ArrayList<String>(8);
 | |
|         for (String filename : expectedNames)
 | |
|         {
 | |
|             check.add(filename);
 | |
|         }
 | |
|         for (FileInfo file : files)
 | |
|         {
 | |
|             if (file.isFolder())
 | |
|             {
 | |
|                 folderCount++;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 fileCount++;
 | |
|             }
 | |
|             check.remove(file.getName());
 | |
|         }
 | |
|         assertTrue("Name list was not exact - remaining: " + check, check.size() == 0);
 | |
|         assertEquals("Incorrect number of files", expectedFileCount, fileCount);
 | |
|         assertEquals("Incorrect number of folders", expectedFolderCount, folderCount);
 | |
|     }
 | |
| 
 | |
|     public void testShallowFilesAndFoldersList() throws Exception
 | |
|     {
 | |
|         List<FileInfo> files = fileFolderService.list(workingRootNodeRef);
 | |
|         // check
 | |
|         String[] expectedNames = new String[]
 | |
|         { NAME_L0_FILE_A, NAME_L0_FILE_B, NAME_L0_FOLDER_A, NAME_L0_FOLDER_B, NAME_L0_FOLDER_C };
 | |
|         checkFileList(files, 2, 3, expectedNames);
 | |
|     }
 | |
| 
 | |
|     public void testShallowFilesAndFoldersListWithLocale() throws Exception
 | |
|     {
 | |
| 		Locale savedLocale = I18NUtil.getContentLocaleOrNull();
 | |
| 		try
 | |
| 		{
 | |
| 			I18NUtil.setContentLocale(Locale.CANADA);
 | |
| 	        List<FileInfo> files = fileFolderService.list(workingRootNodeRef);
 | |
| 	        // check
 | |
| 	        String[] expectedNames = new String[]
 | |
| 	        { NAME_L0_FILE_A, NAME_L0_FILE_B, NAME_L0_FOLDER_A, NAME_L0_FOLDER_B, NAME_L0_FOLDER_C };
 | |
| 	        checkFileList(files, 2, 3, expectedNames);
 | |
| 		}
 | |
| 		finally
 | |
| 		{
 | |
| 	        I18NUtil.setContentLocale(savedLocale);
 | |
| 		}
 | |
|     }
 | |
|     
 | |
|     public void testListPage() throws Exception
 | |
|     {
 | |
|         // sanity checks only (see also GetChildrenCannedQueryTest)
 | |
|         
 | |
|         PagingRequest pagingRequest = new PagingRequest(100, null);
 | |
|         PagingResults<FileInfo> pagingResults = fileFolderService.list(workingRootNodeRef, true, true, null, null, null, pagingRequest);
 | |
|         
 | |
|         assertNotNull(pagingResults);
 | |
|         assertFalse(pagingResults.hasMoreItems());
 | |
|         assertTrue((pagingResults.getQueryExecutionId() != null) && (pagingResults.getQueryExecutionId().length() > 0));
 | |
|         assertNull(pagingResults.getTotalResultCount());
 | |
|         
 | |
|         List<FileInfo> files = pagingResults.getPage();
 | |
|         
 | |
|         // check
 | |
|         String[] expectedNames = new String[]
 | |
|         { NAME_L0_FILE_A, NAME_L0_FILE_B, NAME_L0_FOLDER_A, NAME_L0_FOLDER_B, NAME_L0_FOLDER_C };
 | |
|         checkFileList(files, 2, 3, expectedNames);
 | |
|         
 | |
|         
 | |
|         // empty list if skip count greater than number of results (ALF-7884)
 | |
|         pagingRequest = new PagingRequest(1000, 3, null);
 | |
|         pagingResults = fileFolderService.list(workingRootNodeRef, true, true, null, null, null, pagingRequest);
 | |
|         
 | |
|         assertNotNull(pagingResults);
 | |
|         assertFalse(pagingResults.hasMoreItems());
 | |
|         assertEquals(0, pagingResults.getPage().size());
 | |
|         
 | |
|         // TODO add more here
 | |
|     }
 | |
| 
 | |
|     public void testShallowFilesOnlyList() throws Exception
 | |
|     {
 | |
|         List<FileInfo> files = fileFolderService.listFiles(workingRootNodeRef);
 | |
|         // check
 | |
|         String[] expectedNames = new String[]
 | |
|         { NAME_L0_FILE_A, NAME_L0_FILE_B };
 | |
|         checkFileList(files, 2, 0, expectedNames);
 | |
|     }
 | |
| 
 | |
|     public void testShallowFoldersOnlyList() throws Exception
 | |
|     {
 | |
|         List<FileInfo> files = fileFolderService.listFolders(workingRootNodeRef);
 | |
|         // check
 | |
|         String[] expectedNames = new String[]
 | |
|         { NAME_L0_FOLDER_A, NAME_L0_FOLDER_B, NAME_L0_FOLDER_C };
 | |
|         checkFileList(files, 0, 3, expectedNames);
 | |
|     }
 | |
| 
 | |
|     public void testShallowFileSearch() throws Exception
 | |
|     {
 | |
|         List<FileInfo> files = fileFolderService.search(workingRootNodeRef, NAME_L0_FILE_B, true, false, false);
 | |
|         // check
 | |
|         String[] expectedNames = new String[]
 | |
|         { NAME_L0_FILE_B };
 | |
|         checkFileList(files, 1, 0, expectedNames);
 | |
|     }
 | |
| 
 | |
|     public void testDeepFilesAndFoldersSearch() throws Exception
 | |
|     {
 | |
|         // Seach for pattern <Any char>-<AnyChars>
 | |
|         {
 | |
|             List<FileInfo> files = fileFolderService.search(workingRootNodeRef, "?1-*", true, true, true);
 | |
|             // check
 | |
|             String[] expectedNames = new String[]
 | |
|                { NAME_L1_FOLDER_A, NAME_L1_FOLDER_B, NAME_L1_FILE_A, NAME_L1_FILE_B, NAME_L1_FILE_C };
 | |
|             checkFileList(files, 3, 2, expectedNames);
 | |
|         }
 | |
|         
 | |
|         // Search for a particular file
 | |
|         {
 | |
|             List<FileInfo> files = fileFolderService.search(workingRootNodeRef, NAME_L1_FILE_B, true, true, true);
 | |
|             // check
 | |
|             String[] expectedNames = new String[]
 | |
|                { NAME_L1_FILE_B };
 | |
|             checkFileList(files, 1, 0, expectedNames);
 | |
|         }
 | |
|         
 | |
|         // Search for all files with wildcard
 | |
|         {
 | |
|             List<FileInfo> files = fileFolderService.search(workingRootNodeRef, "*", true, true, true);
 | |
|             // check
 | |
|             String[] expectedNames = new String[]
 | |
|                { 
 | |
|                  NAME_CHECK_FOLDER,       
 | |
|                  NAME_L0_FOLDER_A, 
 | |
|                  NAME_L0_FOLDER_B, 
 | |
|                  NAME_L0_FOLDER_C, 
 | |
|                  NAME_L1_FOLDER_A, 
 | |
|                  NAME_L1_FOLDER_B,
 | |
|                  NAME_CHECK_FILE,
 | |
|                  NAME_L0_FILE_A, 
 | |
|                  NAME_L0_FILE_B, 
 | |
|                  NAME_L1_FILE_A, 
 | |
|                  NAME_L1_FILE_B, 
 | |
|                  NAME_L1_FILE_C 
 | |
|                };
 | |
|             checkFileList(files, 6, 6, expectedNames);
 | |
|         }
 | |
|         
 | |
|     }
 | |
| 
 | |
|     public void testDeepFilesOnlySearch() throws Exception
 | |
|     {
 | |
|         List<FileInfo> files = fileFolderService.search(workingRootNodeRef, "?1-*", true, false, true);
 | |
|         // check
 | |
|         String[] expectedNames = new String[]
 | |
|         { NAME_L1_FILE_A, NAME_L1_FILE_B, NAME_L1_FILE_C };
 | |
|         checkFileList(files, 3, 0, expectedNames);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Helper to fetch a file or folder by name
 | |
|      * 
 | |
|      * @param name the name of the file or folder
 | |
|      * @param isFolder true if we want a folder, otherwise false if we want a file
 | |
|      * @return Returns the info for the file or folder
 | |
|      */
 | |
|     private FileInfo getByName(String name, boolean isFolder) throws Exception
 | |
|     {
 | |
|         List<FileInfo> results = fileFolderService.search(workingRootNodeRef, name, !isFolder, isFolder, true);
 | |
|         if (results.size() > 1)
 | |
|         {
 | |
|             throw new AlfrescoRuntimeException("Name is not unique in hierarchy: \n" + "   name: " + name + "\n"
 | |
|                     + "   is folder: " + isFolder);
 | |
|         }
 | |
|         else if (results.size() == 0)
 | |
|         {
 | |
|             return null;
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             return results.get(0);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Ensure that an internal method is working - it gets used extensively by following tests
 | |
|      * 
 | |
|      * @see #getByName(String, boolean)
 | |
|      */
 | |
|     public void testGetByName() throws Exception
 | |
|     {
 | |
|         FileInfo fileInfo = getByName(NAME_CHECK_FOLDER, true);
 | |
|         assertNotNull(fileInfo);
 | |
|         assertTrue(fileInfo.isFolder());
 | |
| 
 | |
|         fileInfo = getByName(NAME_CHECK_FILE, false);
 | |
|         assertNotNull(fileInfo);
 | |
|         assertFalse(fileInfo.isFolder());
 | |
|     }
 | |
| 
 | |
|     public void testRenameNormal() throws Exception
 | |
|     {
 | |
|         FileInfo folderInfo = getByName(NAME_L0_FOLDER_A, true);
 | |
|         assertNotNull(folderInfo);
 | |
|         // rename normal
 | |
|         String newName = "DUPLICATE - renamed";
 | |
|         folderInfo = fileFolderService.rename(folderInfo.getNodeRef(), newName);
 | |
|         // check it
 | |
|         FileInfo checkInfo = getByName(NAME_L0_FOLDER_A, true);
 | |
|         assertNull("Folder info should have been renamed away", checkInfo);
 | |
|         checkInfo = getByName(newName, true);
 | |
|         assertNotNull("Folder info for new name is not present", checkInfo);
 | |
|     }
 | |
| 
 | |
|     public void testRenameWithoutAssocQNameChange() throws Exception
 | |
|     {
 | |
|         FileInfo folderInfo = getByName(NAME_L0_FOLDER_A, true);
 | |
|         assertNotNull(folderInfo);
 | |
|         NodeRef folderNodeRef = folderInfo.getNodeRef();
 | |
|         // Create a child file
 | |
|         QName assocQName = QName.createQName(NamespaceService.APP_MODEL_1_0_URI, "abc");
 | |
|         NodeRef newFileNodeRef = fileFolderService.create(folderNodeRef, "AnotherFile.txt", ContentModel.TYPE_CONTENT,
 | |
|                 assocQName).getNodeRef();
 | |
|         // Make sure that the correct association QName was used
 | |
|         QName checkQName = nodeService.getPrimaryParent(newFileNodeRef).getQName();
 | |
|         assertEquals("The given assoc QName was not used for the path", assocQName, checkQName);
 | |
|         // Rename
 | |
|         String newName = "AnotherFile-new.txt";
 | |
|         folderInfo = fileFolderService.rename(newFileNodeRef, newName);
 | |
|         // Make sure that the association QName did not change
 | |
|         checkQName = nodeService.getPrimaryParent(newFileNodeRef).getQName();
 | |
|         assertEquals("The given assoc QName was not used for the path after a rename", assocQName, nodeService
 | |
|                 .getPrimaryParent(newFileNodeRef).getQName());
 | |
|     }
 | |
| 
 | |
|     public void testRenameDuplicate() throws Exception
 | |
|     {
 | |
|         FileInfo folderInfo = getByName(NAME_L0_FOLDER_A, true);
 | |
|         assertNotNull(folderInfo);
 | |
|         // rename duplicate. A file with that name already exists
 | |
|         String newName = NAME_L0_FILE_A;
 | |
|         try
 | |
|         {
 | |
|             folderInfo = fileFolderService.rename(folderInfo.getNodeRef(), newName);
 | |
|             fail("Existing file not detected");
 | |
|         }
 | |
|         catch (FileExistsException e)
 | |
|         {
 | |
|             // expected
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     public void testRenameDiscussionALF5569() throws Exception
 | |
|     {
 | |
|         FileInfo fileInfo = getByName(NAME_L0_FILE_A, false);
 | |
|         assertNotNull(fileInfo);
 | |
|         
 | |
|         // create a discussion for the file, this happens in a behaviour
 | |
|         // when adding the discussable aspect
 | |
|         nodeService.addAspect(fileInfo.getNodeRef(), ForumModel.ASPECT_DISCUSSABLE, null);
 | |
|         List<ChildAssociationRef> destChildren = nodeService.getChildAssocs(
 | |
|               fileInfo.getNodeRef(),
 | |
|               ForumModel.ASSOC_DISCUSSION,
 | |
|               RegexQNamePattern.MATCH_ALL);
 | |
|         assertEquals(1, destChildren.size());
 | |
|         
 | |
|         // get the first child
 | |
|         NodeRef discussionNodeRef = destChildren.get(0).getChildRef();
 | |
|         
 | |
|         // check the current name
 | |
|         String currentName = (String)nodeService.getProperty(discussionNodeRef, ContentModel.PROP_NAME);
 | |
|         assertFalse(NAME_DISCUSSION_FOLDER.equals(currentName));
 | |
|         
 | |
|         // rename the discussion node
 | |
|         FileInfo newFileInfo = fileFolderService.rename(discussionNodeRef, NAME_DISCUSSION_FOLDER);
 | |
|         
 | |
|         // get the name now
 | |
|         String newName = (String)nodeService.getProperty(newFileInfo.getNodeRef(), ContentModel.PROP_NAME);
 | |
|         assertEquals(NAME_DISCUSSION_FOLDER, newName);
 | |
|     }
 | |
| 
 | |
|     public void testMove() throws Exception
 | |
|     {
 | |
|         // we are testing failures as well
 | |
|         txn.commit();
 | |
|         // start a new one
 | |
|         txn = transactionService.getNonPropagatingUserTransaction();
 | |
|         txn.begin();
 | |
|         
 | |
|         FileInfo folderToMoveInfo = getByName(NAME_L1_FOLDER_A, true);
 | |
|         assertNotNull(folderToMoveInfo);
 | |
|         NodeRef folderToMoveRef = folderToMoveInfo.getNodeRef();
 | |
|         // move it to the root
 | |
|         fileFolderService.move(folderToMoveRef, workingRootNodeRef, null);
 | |
|         // make sure that it is an immediate child of the root
 | |
|         List<FileInfo> checkFileInfos = fileFolderService.search(workingRootNodeRef, NAME_L1_FOLDER_A, false);
 | |
|         assertEquals("Folder not moved to root", 1, checkFileInfos.size());
 | |
|         // rename properly
 | |
|         FileInfo checkFileInfo = fileFolderService.move(folderToMoveRef, null, "new name");
 | |
|         checkFileInfos = fileFolderService.search(workingRootNodeRef, checkFileInfo.getName(), false);
 | |
|         assertEquals("Folder not renamed in root", 1, checkFileInfos.size());
 | |
|         // attempt illegal rename (existing)
 | |
|         try
 | |
|         {
 | |
|             fileFolderService.move(folderToMoveRef, null, NAME_L0_FOLDER_A);
 | |
|             fail("Existing folder not detected");
 | |
|         }
 | |
|         catch (FileExistsException e)
 | |
|         {
 | |
|             // expected
 | |
|         }
 | |
|         
 | |
|         txn.rollback();
 | |
|         txn = transactionService.getNonPropagatingUserTransaction();
 | |
|         txn.begin();
 | |
| 
 | |
|         // Move a file to a new location
 | |
|         FileInfo fileA = getByName(NAME_L1_FILE_A, false);
 | |
|         FileInfo folderB = getByName(NAME_L0_FOLDER_B, true);
 | |
|         fileFolderService.copy(fileA.getNodeRef(), folderB.getNodeRef(), null);
 | |
|         try
 | |
|         {
 | |
|             // Move to a target folder without a rename and expecting a name clash
 | |
|             fileFolderService.move(fileA.getNodeRef(), folderB.getNodeRef(), null);
 | |
|             fail("Duplicately-named file in target folder was not detected");
 | |
|         }
 | |
|         catch (FileExistsException e)
 | |
|         {
 | |
|             // Expected
 | |
|         }
 | |
| 
 | |
|         txn.rollback();
 | |
|         txn = transactionService.getNonPropagatingUserTransaction();
 | |
|         txn.begin();
 | |
| 
 | |
|         // Move to a target folder but with a rename to avoid the name clash
 | |
|         fileFolderService.move(fileA.getNodeRef(), folderB.getNodeRef(), NAME_L1_FILE_B);
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * <a href="https://issues.alfresco.com/jira/browse/ALF-7692">ALF-7692</a>
 | |
|      */
 | |
|     public void testMovePermissions() throws Exception
 | |
|     {
 | |
|         txn.commit();
 | |
|         
 | |
|         // Create a target folder to write to.  Folder owner is 'system'.
 | |
|         RunAsWork<NodeRef> createTargetWork = new RunAsWork<NodeRef>()
 | |
|         {
 | |
|             @Override
 | |
|             public NodeRef doWork() throws Exception
 | |
|             {
 | |
|                 // Create folder TARGET
 | |
|                 return fileFolderService.create(
 | |
|                         workingRootNodeRef,
 | |
|                         "TARGET",
 | |
|                         ContentModel.TYPE_FOLDER).getNodeRef();
 | |
|             }
 | |
|         };
 | |
|         final NodeRef targetNodeRef = AuthenticationUtil.runAs(createTargetWork, AuthenticationUtil.getSystemUserName());
 | |
| 
 | |
|         // Use a specific user
 | |
|         String username = "Mover-" + GUID.generate();
 | |
|         char[] password = "mover".toCharArray();
 | |
|         authenticationService.createAuthentication(username, password);
 | |
|         permissionService.setPermission(
 | |
|                 rootNodeRef,
 | |
|                 username,
 | |
|                 PermissionService.ALL_PERMISSIONS,
 | |
|                 true);
 | |
|         AuthenticationUtil.clearCurrentSecurityContext();
 | |
|         AuthenticationUtil.setFullyAuthenticatedUser(username);
 | |
| 
 | |
|         // Check that we can write to the target while permissions allow it
 | |
|         fileFolderService.create(
 | |
|                 targetNodeRef,
 | |
|                 "SOURCE ONE",
 | |
|                 ContentModel.TYPE_CONTENT).getNodeRef();
 | |
|         // Deny anyone access to the target
 | |
|         RunAsWork<Void> setPermissionsWork = new RunAsWork<Void>()
 | |
|         {
 | |
|             @Override
 | |
|             public Void doWork() throws Exception
 | |
|             {
 | |
|                 permissionService.setInheritParentPermissions(targetNodeRef, false);
 | |
|                 permissionService.setPermission(
 | |
|                         targetNodeRef,
 | |
|                         PermissionService.ALL_AUTHORITIES,
 | |
|                         PermissionService.ALL_PERMISSIONS,
 | |
|                         false);
 | |
|                 return null;
 | |
|             }
 | |
|         };
 | |
|         AuthenticationUtil.runAs(setPermissionsWork, AuthenticationUtil.getSystemUserName());
 | |
|         try
 | |
|         {
 | |
|             fileFolderService.create(
 | |
|                     targetNodeRef,
 | |
|                     "SOURCE TWO",
 | |
|                     ContentModel.TYPE_CONTENT).getNodeRef();
 | |
|             fail("Expected permissions to deny a write");
 | |
|         }
 | |
|         catch (AccessDeniedException e)
 | |
|         {
 | |
|             // Expected
 | |
|         }
 | |
| 
 | |
|         // Create source to move
 | |
|         NodeRef movingNodeRef = fileFolderService.create(
 | |
|                 workingRootNodeRef,
 | |
|                 "SOURCE THREE",
 | |
|                 ContentModel.TYPE_CONTENT).getNodeRef();
 | |
|         // Move it
 | |
|         try
 | |
|         {
 | |
|             fileFolderService.moveFrom(movingNodeRef, workingRootNodeRef, targetNodeRef, "SOURCE THREE");
 | |
|             fail("Expected permissions to deny the move");
 | |
|         }
 | |
|         catch (AccessDeniedException e)
 | |
|         {
 | |
|             // Expected
 | |
|         }
 | |
|         // Move it
 | |
|         try
 | |
|         {
 | |
|             fileFolderService.move(movingNodeRef, targetNodeRef, "SOURCE FOUR");
 | |
|             fail("Expected permissions to deny the move");
 | |
|         }
 | |
|         catch (AccessDeniedException e)
 | |
|         {
 | |
|             // Expected
 | |
|         }
 | |
|         // Copy it
 | |
|         try
 | |
|         {
 | |
|             fileFolderService.copy(movingNodeRef, targetNodeRef, "SOURCE FIVE");
 | |
|             fail("Expected permissions to deny the copy");
 | |
|         }
 | |
|         catch (AccessDeniedException e)
 | |
|         {
 | |
|             // Expected
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public void testCopy() throws Exception
 | |
|     {
 | |
|         FileInfo folderToCopyInfo = getByName(NAME_L1_FOLDER_A, true);
 | |
|         assertNotNull(folderToCopyInfo);
 | |
|         NodeRef folderToCopyRef = folderToCopyInfo.getNodeRef();
 | |
|         // copy it to the root
 | |
|         folderToCopyInfo = fileFolderService.copy(folderToCopyRef, workingRootNodeRef, null);
 | |
|         folderToCopyRef = folderToCopyInfo.getNodeRef();
 | |
|         // make sure that it is an immediate child of the root
 | |
|         List<FileInfo> checkFileInfos = fileFolderService.search(workingRootNodeRef, NAME_L1_FOLDER_A, false);
 | |
|         assertEquals("Folder not copied to root", 1, checkFileInfos.size());
 | |
|         // copy properly
 | |
|         FileInfo checkFileInfo = fileFolderService.copy(folderToCopyRef, null, "new name");
 | |
|         checkFileInfos = fileFolderService.search(workingRootNodeRef, checkFileInfo.getName(), false);
 | |
|         assertEquals("Folder not renamed in root", 1, checkFileInfos.size());
 | |
|         // attempt illegal copy (existing)
 | |
|         try
 | |
|         {
 | |
|             fileFolderService.copy(folderToCopyRef, null, NAME_L0_FOLDER_A);
 | |
|             fail("Existing folder not detected");
 | |
|         }
 | |
|         catch (FileExistsException e)
 | |
|         {
 | |
|             // expected
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public void testCreateFolder() throws Exception
 | |
|     {
 | |
|         // we are testing failures as well
 | |
|         txn.commit();
 | |
|         // start a new one
 | |
|         txn = transactionService.getNonPropagatingUserTransaction();
 | |
|         txn.begin();
 | |
| 
 | |
|         FileInfo parentFolderInfo = getByName(NAME_L0_FOLDER_A, true);
 | |
|         assertNotNull(parentFolderInfo);
 | |
|         NodeRef parentFolderRef = parentFolderInfo.getNodeRef();
 | |
|         // create a file that already exists
 | |
|         UserTransaction rollbackTxn = null;
 | |
|         try
 | |
|         {
 | |
|             rollbackTxn = transactionService.getNonPropagatingUserTransaction();
 | |
|             rollbackTxn.begin();
 | |
|             fileFolderService.create(parentFolderRef, NAME_L1_FILE_A, ContentModel.TYPE_CONTENT);
 | |
|             fail("Failed to detect duplicate filename");
 | |
|         }
 | |
|         catch (FileExistsException e)
 | |
|         {
 | |
|             // expected
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             rollbackTxn.rollback();
 | |
|         }
 | |
|         // create folder of illegal type
 | |
|         try
 | |
|         {
 | |
|             rollbackTxn = transactionService.getNonPropagatingUserTransaction();
 | |
|             rollbackTxn.begin();
 | |
|             fileFolderService.create(parentFolderRef, "illegal folder", ContentModel.TYPE_SYSTEM_FOLDER);
 | |
|             fail("Illegal type not detected");
 | |
|         }
 | |
|         catch (RuntimeException e)
 | |
|         {
 | |
|             // expected
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             rollbackTxn.rollback();
 | |
|         }
 | |
|         // Create a cm:folder derived type
 | |
|         try
 | |
|         {
 | |
|             rollbackTxn = transactionService.getNonPropagatingUserTransaction();
 | |
|             rollbackTxn.begin();
 | |
|             // Create a new model
 | |
|             String testNs = "http://www.alfresco.org/model/test111/1.0";
 | |
|             M2Model testModel = M2Model.createModel("t111:filefolderserviceimpltest");
 | |
|             testModel.createNamespace(testNs, "t111");
 | |
|             testModel.createImport(NamespaceService.DICTIONARY_MODEL_1_0_URI, NamespaceService.DICTIONARY_MODEL_PREFIX);
 | |
|             testModel.createImport(NamespaceService.SYSTEM_MODEL_1_0_URI, NamespaceService.SYSTEM_MODEL_PREFIX);
 | |
|             testModel.createImport(NamespaceService.CONTENT_MODEL_1_0_URI, NamespaceService.CONTENT_MODEL_PREFIX);
 | |
| 
 | |
|             M2Type testType = testModel.createType("t111:subfolder");
 | |
|             testType.setParentName("cm:" + ContentModel.TYPE_FOLDER.getLocalName());
 | |
|             dictionaryDAO.putModel(testModel);
 | |
|             fileFolderService
 | |
|                     .create(parentFolderRef, "Legal subtype of folder", QName.createQName(testNs, "subfolder"));
 | |
|         }
 | |
|         catch (Throwable e)
 | |
|         {
 | |
|             throw new Exception("Legal subtype of cm:folder not allowed.", e);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             rollbackTxn.rollback();
 | |
|         }
 | |
| 
 | |
|         // create a file
 | |
|         FileInfo fileInfo = fileFolderService.create(parentFolderRef, "newFile", ContentModel.TYPE_CONTENT);
 | |
|         // check
 | |
|         assertTrue("Node not created", nodeService.exists(fileInfo.getNodeRef()));
 | |
|         assertFalse("File type expected", fileInfo.isFolder());
 | |
|     }
 | |
| 
 | |
|     public void testCreateFile() throws Exception
 | |
|     {
 | |
| 
 | |
|     }
 | |
| 
 | |
|     public void testCreateInRoot() throws Exception
 | |
|     {
 | |
|         fileFolderService.create(rootNodeRef, "New Folder", ContentModel.TYPE_FOLDER);
 | |
|     }
 | |
| 
 | |
|     public void testMakeFolders() throws Exception
 | |
|     {
 | |
|         // create a completely new path below the root
 | |
|         List<String> namePath = new ArrayList<String>(4);
 | |
|         namePath.add("AAA");
 | |
|         namePath.add("BBB");
 | |
|         namePath.add("CCC");
 | |
|         namePath.add("DDD");
 | |
| 
 | |
|         FileInfo lastFileInfo = FileFolderServiceImpl.makeFolders(fileFolderService, rootNodeRef, namePath,
 | |
|                 ContentModel.TYPE_FOLDER);
 | |
|         assertNotNull("First makeFolder failed", lastFileInfo);
 | |
|         // check that a repeat works
 | |
| 
 | |
|         FileInfo lastFileInfoAgain = FileFolderServiceImpl.makeFolders(fileFolderService, rootNodeRef, namePath,
 | |
|                 ContentModel.TYPE_FOLDER);
 | |
|         assertNotNull("Repeat makeFolders failed", lastFileInfoAgain);
 | |
|         assertEquals("Repeat created new leaf", lastFileInfo.getNodeRef(), lastFileInfoAgain.getNodeRef());
 | |
|         // check that it worked
 | |
|         List<FileInfo> checkInfos = fileFolderService.search(rootNodeRef, "DDD", false, true, true);
 | |
|         assertEquals("Expected to find a result", 1, checkInfos.size());
 | |
|         // get the path
 | |
|         List<FileInfo> checkPathInfos = fileFolderService.getNamePath(rootNodeRef, checkInfos.get(0).getNodeRef());
 | |
|         assertEquals("Path created is incorrect", namePath.size(), checkPathInfos.size());
 | |
|         int i = 0;
 | |
|         for (FileInfo checkInfo : checkPathInfos)
 | |
|         {
 | |
|             assertEquals("Path mismatch", namePath.get(i), checkInfo.getName());
 | |
|             i++;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Lucene only indexes terms that are 3 characters or more
 | |
|      */
 | |
|     public void testMakeFoldersShortNames() throws Exception
 | |
|     {
 | |
|         // create a completely new path below the root
 | |
|         List<String> namePath = new ArrayList<String>(4);
 | |
|         namePath.add("A");
 | |
|         namePath.add("B");
 | |
|         namePath.add("C");
 | |
|         namePath.add("D");
 | |
| 
 | |
|         FileInfo lastFileInfo = FileFolderServiceImpl.makeFolders(fileFolderService, rootNodeRef, namePath,
 | |
|                 ContentModel.TYPE_FOLDER);
 | |
|         assertNotNull("First makeFolder failed", lastFileInfo);
 | |
|         // check that a repeat works
 | |
| 
 | |
|         FileInfo lastFileInfoAgain = FileFolderServiceImpl.makeFolders(fileFolderService, rootNodeRef, namePath,
 | |
|                 ContentModel.TYPE_FOLDER);
 | |
|         assertNotNull("Repeat makeFolders failed", lastFileInfoAgain);
 | |
|         assertEquals("Repeat created new leaf", lastFileInfo.getNodeRef(), lastFileInfoAgain.getNodeRef());
 | |
|     }
 | |
| 
 | |
|     public void testGetNamePath() throws Exception
 | |
|     {
 | |
|         FileInfo fileInfo = getByName(NAME_L1_FILE_A, false);
 | |
|         assertNotNull(fileInfo);
 | |
|         NodeRef nodeRef = fileInfo.getNodeRef();
 | |
| 
 | |
|         List<FileInfo> infoPaths = fileFolderService.getNamePath(workingRootNodeRef, nodeRef);
 | |
|         assertEquals("Not enough elements", 2, infoPaths.size());
 | |
|         assertEquals("First level incorrent", NAME_L0_FOLDER_A, infoPaths.get(0).getName());
 | |
|         assertEquals("Second level incorrent", NAME_L1_FILE_A, infoPaths.get(1).getName());
 | |
| 
 | |
|         // pass in a null root and make sure that it still works
 | |
|         infoPaths = fileFolderService.getNamePath(null, nodeRef);
 | |
|         assertEquals("Not enough elements", 3, infoPaths.size());
 | |
|         assertEquals("First level incorrent", workingRootNodeRef.getId(), infoPaths.get(0).getName());
 | |
|         assertEquals("Second level incorrent", NAME_L0_FOLDER_A, infoPaths.get(1).getName());
 | |
|         assertEquals("Third level incorrent", NAME_L1_FILE_A, infoPaths.get(2).getName());
 | |
| 
 | |
|         // check that a non-aligned path is detected
 | |
|         NodeRef startRef = getByName(NAME_L0_FOLDER_B, true).getNodeRef();
 | |
|         try
 | |
|         {
 | |
|             fileFolderService.getNamePath(startRef, nodeRef);
 | |
|             fail("Failed to detect non-aligned path from root to target node");
 | |
|         }
 | |
|         catch (FileNotFoundException e)
 | |
|         {
 | |
|             // expected
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public void testSearchSimple() throws Exception
 | |
|     {
 | |
|         FileInfo folderInfo = getByName(NAME_L0_FOLDER_A, true);
 | |
|         assertNotNull(folderInfo);
 | |
|         NodeRef folderNodeRef = folderInfo.getNodeRef();
 | |
|         // search for a file that is not there
 | |
|         NodeRef phantomNodeRef = fileFolderService.searchSimple(folderNodeRef, "aaaaaaa");
 | |
|         assertNull("Found non-existent node by name", phantomNodeRef);
 | |
|         // search for a file that is there
 | |
|         NodeRef fileNodeRef = fileFolderService.searchSimple(folderNodeRef, NAME_L1_FILE_A);
 | |
|         assertNotNull("Didn't find file", fileNodeRef);
 | |
|         // double check
 | |
|         FileInfo checkInfo = getByName(NAME_L1_FILE_A, false);
 | |
|         assertEquals("Incorrect node found", checkInfo.getNodeRef(), fileNodeRef);
 | |
|     }
 | |
| 
 | |
|     public void testResolveNamePath() throws Exception
 | |
|     {
 | |
|         FileInfo fileInfo = getByName(NAME_L1_FILE_A, false);
 | |
|         List<String> pathElements = new ArrayList<String>(3);
 | |
|         pathElements.add(NAME_L0_FOLDER_A);
 | |
|         pathElements.add(NAME_L1_FILE_A);
 | |
| 
 | |
|         FileInfo fileInfoCheck = fileFolderService.resolveNamePath(workingRootNodeRef, pathElements);
 | |
|         assertNotNull("File info not found", fileInfoCheck);
 | |
|         assertEquals("Path not resolved to correct node", fileInfo.getNodeRef(), fileInfoCheck.getNodeRef());
 | |
|     }
 | |
| 
 | |
|     public void testGetReaderWriter() throws Exception
 | |
|     {
 | |
|         // testing a failure
 | |
|         txn.commit();
 | |
|         txn = transactionService.getUserTransaction();
 | |
|         txn.begin();
 | |
| 
 | |
|         FileInfo dirInfo = getByName(NAME_L0_FOLDER_A, true);
 | |
| 
 | |
|         UserTransaction rollbackTxn = null;
 | |
|         try
 | |
|         {
 | |
|             rollbackTxn = transactionService.getNonPropagatingUserTransaction();
 | |
|             rollbackTxn.begin();
 | |
|             fileFolderService.getWriter(dirInfo.getNodeRef());
 | |
|             fail("Failed to detect content write to folder");
 | |
|         }
 | |
|         catch (RuntimeException e)
 | |
|         {
 | |
|             // expected
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             rollbackTxn.rollback();
 | |
|         }
 | |
| 
 | |
|         FileInfo fileInfo = getByName(NAME_L1_FILE_A, false);
 | |
| 
 | |
|         ContentWriter writer = fileFolderService.getWriter(fileInfo.getNodeRef());
 | |
|         assertNotNull("Writer is null", writer);
 | |
|         // write some content
 | |
|         String content = "ABC";
 | |
|         writer.putContent(content);
 | |
|         // read the content
 | |
|         ContentReader reader = fileFolderService.getReader(fileInfo.getNodeRef());
 | |
|         assertNotNull("Reader is null", reader);
 | |
|         String checkContent = reader.getContentString();
 | |
|         assertEquals("Content mismatch", content, checkContent);
 | |
|     }
 | |
| 
 | |
|     public void testLongFileNames() throws Exception
 | |
|     {
 | |
|         String fileName = "12345678901234567890123456789012345678901234567890"
 | |
|                 + "12345678901234567890123456789012345678901234567890"
 | |
|                 + "12345678901234567890123456789012345678901234567890"
 | |
|                 + "12345678901234567890123456789012345678901234567890"
 | |
|                 + "12345678901234567890123456789012345678901234567890"
 | |
|                 + "12345678901234567890123456789012345678901234567890";
 | |
|         FileInfo fileInfo = fileFolderService.create(workingRootNodeRef, fileName, ContentModel.TYPE_CONTENT);
 | |
|         // see if we can get it again
 | |
|         NodeRef fileNodeRef = fileFolderService.searchSimple(workingRootNodeRef, fileName);
 | |
|         assertNotNull("Long filename not found", fileNodeRef);
 | |
|         assertEquals(fileInfo.getNodeRef(), fileNodeRef);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Validates <a href="https://issues.alfresco.com/jira/browse/ALFCOM-2655">ACT-7225</a>
 | |
|      */
 | |
|     public void testGetType() throws Exception
 | |
|     {
 | |
|     	Locale savedLocale = I18NUtil.getContentLocaleOrNull();
 | |
|     	try
 | |
|     	{
 | |
| 	        I18NUtil.setContentLocale(Locale.CANADA);
 | |
| 	        FileFolderServiceType type = fileFolderService.getType(ContentModel.TYPE_FOLDER);
 | |
| 	        assertEquals("Type incorrect for folder", FileFolderServiceType.FOLDER, type);
 | |
|     	}
 | |
|     	finally
 | |
|     	{
 | |
|             I18NUtil.setContentLocale(savedLocale);
 | |
|     	}
 | |
|     }
 | |
|     
 | |
|     public void testETHREEOH_3088_MoveIntoSelf() throws Exception
 | |
|     {
 | |
|         FileInfo folderInfo = fileFolderService.create(workingRootNodeRef, "NotGood.txt", ContentModel.TYPE_FOLDER);
 | |
|         NodeRef folderNodeRef = folderInfo.getNodeRef();
 | |
|         // Move into self
 | |
|         try
 | |
|         {
 | |
|             fileFolderService.move(folderNodeRef, folderNodeRef, null);
 | |
|             fail("Failed to detect cyclic relationship");
 | |
|         }
 | |
|         catch (CyclicChildRelationshipException e)
 | |
|         {
 | |
|             // Expected
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     public void testAlf6560MimetypeSetting() throws Exception
 | |
|     {
 | |
|         FileInfo fileInfo = fileFolderService.create(workingRootNodeRef, "Something.html", ContentModel.TYPE_CONTENT);
 | |
|         NodeRef fileNodeRef = fileInfo.getNodeRef();
 | |
|         
 | |
|         // Write the content but without setting the mimetype
 | |
|         ContentWriter writer = fileFolderService.getWriter(fileNodeRef);
 | |
|         writer.putContent("CONTENT");
 | |
|         
 | |
|         ContentReader reader = fileFolderService.getReader(fileNodeRef);
 | |
|         assertEquals("Mimetype was not automatically set", MimetypeMap.MIMETYPE_HTML, reader.getMimetype());
 | |
|         
 | |
|         
 | |
|         // Now ask for encoding too
 | |
|         writer = fileFolderService.getWriter(fileNodeRef);
 | |
|         writer.guessEncoding();
 | |
|         OutputStream out = writer.getContentOutputStream();
 | |
|         out.write( "<html><body>hall\u00e5 v\u00e4rlden</body></html>".getBytes("UnicodeBig") );
 | |
|         out.close();
 | |
|         
 | |
|         reader = fileFolderService.getReader(fileNodeRef);
 | |
|         assertEquals("Mimetype was not automatically set", MimetypeMap.MIMETYPE_HTML, reader.getMimetype());
 | |
|         assertEquals("Encoding was not automatically set", "UTF-16BE", reader.getEncoding());
 | |
|     }
 | |
| 
 | |
|     @SuppressWarnings("unused")
 | |
|     public void testGetLocalizedSibling() throws Exception
 | |
|     {
 | |
|         FileInfo base = fileFolderService.create(workingRootNodeRef, "Something.ftl", ContentModel.TYPE_CONTENT);
 | |
|         NodeRef node = base.getNodeRef();
 | |
|         NodeRef nodeFr = fileFolderService.create(workingRootNodeRef, "Something_fr.ftl", ContentModel.TYPE_CONTENT).getNodeRef();
 | |
|         NodeRef nodeFrFr = fileFolderService.create(workingRootNodeRef, "Something_fr_FR..ftl", ContentModel.TYPE_CONTENT).getNodeRef();
 | |
|         NodeRef nodeEn = fileFolderService.create(workingRootNodeRef, "Something_en.ftl", ContentModel.TYPE_CONTENT).getNodeRef();
 | |
|         NodeRef nodeEnUs = fileFolderService.create(workingRootNodeRef, "Something_en_US.ftl", ContentModel.TYPE_CONTENT).getNodeRef();
 | |
|         
 | |
|         I18NUtil.setLocale(Locale.US);
 | |
|         assertEquals("Match fail for " + I18NUtil.getLocale(), nodeEnUs, fileFolderService.getLocalizedSibling(node));
 | |
|         I18NUtil.setLocale(Locale.UK);
 | |
|         assertEquals("Match fail for " + I18NUtil.getLocale(), nodeEn, fileFolderService.getLocalizedSibling(node));
 | |
|         I18NUtil.setLocale(Locale.CHINESE);
 | |
|         assertEquals("Match fail for " + I18NUtil.getLocale(), node, fileFolderService.getLocalizedSibling(node));
 | |
| 
 | |
|         // Now use French as the base and check that the original is returned
 | |
|         
 | |
|         I18NUtil.setLocale(Locale.US);
 | |
|         assertEquals("Match fail for " + I18NUtil.getLocale(), nodeFr, fileFolderService.getLocalizedSibling(nodeFr));
 | |
|         I18NUtil.setLocale(Locale.UK);
 | |
|         assertEquals("Match fail for " + I18NUtil.getLocale(), nodeFr, fileFolderService.getLocalizedSibling(nodeFr));
 | |
|         I18NUtil.setLocale(Locale.CHINESE);
 | |
|         assertEquals("Match fail for " + I18NUtil.getLocale(), nodeFr, fileFolderService.getLocalizedSibling(nodeFr));
 | |
|         
 | |
|         
 | |
|         // Check that extensions like .get.html.ftl work
 | |
|         FileInfo mbase = fileFolderService.create(workingRootNodeRef, "Another.get.html.ftl", ContentModel.TYPE_CONTENT);
 | |
|         NodeRef mnode = mbase.getNodeRef();
 | |
|         NodeRef mnodeFr = fileFolderService.create(workingRootNodeRef, "Another_fr.get.html.ftl", ContentModel.TYPE_CONTENT).getNodeRef();
 | |
|         
 | |
|         // Should get the base version, except for when French
 | |
|         I18NUtil.setLocale(Locale.UK);
 | |
|         assertEquals("Match fail for " + I18NUtil.getLocale(), mnode, fileFolderService.getLocalizedSibling(mnode));
 | |
|         I18NUtil.setLocale(Locale.FRENCH);
 | |
|         assertEquals("Match fail for " + I18NUtil.getLocale(), mnodeFr, fileFolderService.getLocalizedSibling(mnode));
 | |
|         I18NUtil.setLocale(Locale.CHINESE);
 | |
|         assertEquals("Match fail for " + I18NUtil.getLocale(), mnode, fileFolderService.getLocalizedSibling(mnode));
 | |
|         I18NUtil.setLocale(Locale.US);
 | |
|         assertEquals("Match fail for " + I18NUtil.getLocale(), mnode, fileFolderService.getLocalizedSibling(mnode));
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Ensures that timestamp propagation can be successfully enabled.<br/>
 | |
|      * <a href="https://issues.alfresco.com/jira/browse/ALF-7421">ALF-7421</a>
 | |
|      */
 | |
|     public synchronized void testAlf7421TimestampPropagation() throws Exception
 | |
|     {
 | |
|         // Terminate the transaction
 | |
|         txn.commit();
 | |
|         
 | |
|         nodeService.addAspect(workingRootNodeRef, ContentModel.ASPECT_AUDITABLE, null);
 | |
|         
 | |
|         FileInfo folderInfo = fileFolderService.create(workingRootNodeRef, "SomeFolder", ContentModel.TYPE_FOLDER);
 | |
|         NodeRef folderNodeRef = folderInfo.getNodeRef();
 | |
|         // Get the dates for the folder we are using
 | |
|         String creatorExpected = (String) nodeService.getProperty(folderNodeRef, ContentModel.PROP_CREATOR);
 | |
|         Date createdExpected = (Date) nodeService.getProperty(folderNodeRef, ContentModel.PROP_CREATED);
 | |
|         String modifierExpected = (String) nodeService.getProperty(folderNodeRef, ContentModel.PROP_MODIFIER);
 | |
|         Date modifiedExpected = (Date) nodeService.getProperty(folderNodeRef, ContentModel.PROP_MODIFIED);
 | |
|         
 | |
|         // Get the current dates for the parent folder (one level up)
 | |
|         String creatorTooHigh = (String) nodeService.getProperty(workingRootNodeRef, ContentModel.PROP_CREATOR);
 | |
|         Date createdTooHigh = (Date) nodeService.getProperty(workingRootNodeRef, ContentModel.PROP_CREATED);
 | |
|         String modifierTooHigh = (String) nodeService.getProperty(workingRootNodeRef, ContentModel.PROP_MODIFIER);
 | |
|         Date modifiedTooHigh = (Date) nodeService.getProperty(workingRootNodeRef, ContentModel.PROP_MODIFIED);
 | |
| 
 | |
|         // Create a new file and check the parent (expect changes)
 | |
|         FileInfo fileInfo = fileFolderService.create(folderNodeRef, "Something.html", ContentModel.TYPE_CONTENT);
 | |
|         NodeRef fileNodeRef = fileInfo.getNodeRef();
 | |
|         nodeService.addAspect(fileNodeRef, ContentModel.ASPECT_AUDITABLE, null);
 | |
|         
 | |
|         assertEquals("cm:creator should not have changed",
 | |
|                 creatorExpected,
 | |
|                 nodeService.getProperty(folderNodeRef, ContentModel.PROP_CREATOR));
 | |
|         assertEquals("cm:created should not have changed",
 | |
|                 createdExpected,
 | |
|                 nodeService.getProperty(folderNodeRef, ContentModel.PROP_CREATED));
 | |
|         assertEquals("cm:modifier should have changed",
 | |
|                 nodeService.getProperty(fileNodeRef, ContentModel.PROP_MODIFIER),
 | |
|                 nodeService.getProperty(folderNodeRef, ContentModel.PROP_MODIFIER));
 | |
|         assertEquals("cm:modified should have changed",
 | |
|                 nodeService.getProperty(fileNodeRef, ContentModel.PROP_MODIFIED),
 | |
|                 nodeService.getProperty(folderNodeRef, ContentModel.PROP_MODIFIED));
 | |
|         // Rename the child and check parent (expect changes)
 | |
|         fileFolderService.rename(fileNodeRef, "something.html");
 | |
|         assertEquals("cm:creator should not have changed",
 | |
|                 creatorExpected,
 | |
|                 nodeService.getProperty(folderNodeRef, ContentModel.PROP_CREATOR));
 | |
|         assertEquals("cm:created should not have changed",
 | |
|                 createdExpected,
 | |
|                 nodeService.getProperty(folderNodeRef, ContentModel.PROP_CREATED));
 | |
|         assertEquals("cm:modifier should not have changed",
 | |
|                 modifierExpected,
 | |
|                 nodeService.getProperty(folderNodeRef, ContentModel.PROP_MODIFIER));
 | |
|         assertNotSame("cm:modified should have changed",
 | |
|                 modifiedExpected,
 | |
|                 nodeService.getProperty(folderNodeRef, ContentModel.PROP_MODIFIED));
 | |
|         // Update the child and check parent (expect NO changes)
 | |
|         modifiedExpected = (Date) nodeService.getProperty(folderNodeRef, ContentModel.PROP_MODIFIED);
 | |
|         nodeService.setProperty(fileNodeRef, ContentModel.PROP_TITLE, "Hippo");
 | |
|         assertEquals("cm:creator should not have changed",
 | |
|                 creatorExpected,
 | |
|                 nodeService.getProperty(folderNodeRef, ContentModel.PROP_CREATOR));
 | |
|         assertEquals("cm:created should not have changed",
 | |
|                 createdExpected,
 | |
|                 nodeService.getProperty(folderNodeRef, ContentModel.PROP_CREATED));
 | |
|         assertEquals("cm:modifier should not have changed",
 | |
|                 modifierExpected,
 | |
|                 nodeService.getProperty(folderNodeRef, ContentModel.PROP_MODIFIER));
 | |
|         assertEquals("cm:modified should not have changed",
 | |
|                 modifiedExpected,
 | |
|                 nodeService.getProperty(folderNodeRef, ContentModel.PROP_MODIFIED));
 | |
|         // Delete node and check parent (expect modifier changes)
 | |
|         fileFolderService.delete(fileNodeRef);
 | |
|         assertEquals("cm:creator should not have changed",
 | |
|                 creatorExpected,
 | |
|                 nodeService.getProperty(folderNodeRef, ContentModel.PROP_CREATOR));
 | |
|         assertEquals("cm:created should not have changed",
 | |
|                 createdExpected,
 | |
|                 nodeService.getProperty(folderNodeRef, ContentModel.PROP_CREATED));
 | |
|         assertSame("cm:modifier should have changed",
 | |
|                 modifierExpected,
 | |
|                 nodeService.getProperty(folderNodeRef, ContentModel.PROP_MODIFIER));
 | |
|         assertNotSame("cm:modified should have changed",
 | |
|                 modifiedExpected,
 | |
|                 nodeService.getProperty(folderNodeRef, ContentModel.PROP_MODIFIED));               
 | |
|         // Finally check that the second level up was NOT modified
 | |
|         assertEquals("cm:creator should not have changed (level too high)",
 | |
|                 creatorTooHigh,
 | |
|                 nodeService.getProperty(workingRootNodeRef, ContentModel.PROP_CREATOR));
 | |
|         assertEquals("cm:created should not have changed (level too high)",
 | |
|                 createdTooHigh,
 | |
|                 nodeService.getProperty(workingRootNodeRef, ContentModel.PROP_CREATED));
 | |
|         assertEquals("cm:modifier should not have changed (level too high)",
 | |
|                 modifierTooHigh,
 | |
|                 nodeService.getProperty(workingRootNodeRef, ContentModel.PROP_MODIFIER));
 | |
|         assertEquals("cm:modified should not have changed (level too high)",
 | |
|                 modifiedTooHigh,
 | |
|                 nodeService.getProperty(workingRootNodeRef, ContentModel.PROP_MODIFIED));
 | |
|     }
 | |
|     
 | |
|     public void testPatterns()
 | |
|     {
 | |
|         // sanity checks only (see also GetChildrenCannedQueryTest)
 | |
|         
 | |
|     	I18NUtil.setContentLocale(Locale.CANADA);
 | |
| 
 | |
|         // test 1
 | |
|         PagingRequest pagingRequest = new PagingRequest(100, null);
 | |
|         PagingResults<FileInfo> pagingResults = fileFolderService.list(workingRootNodeRef, true, true, "L0*", null, null, pagingRequest);
 | |
|         
 | |
|         assertNotNull(pagingResults);
 | |
|         assertFalse(pagingResults.hasMoreItems());
 | |
|         assertNull(pagingResults.getTotalResultCount());
 | |
|         
 | |
|         List<FileInfo> files = pagingResults.getPage();
 | |
|         
 | |
|         // check
 | |
|         String[] expectedNames = new String[]
 | |
|         { NAME_L0_FILE_A, NAME_L0_FILE_B, NAME_L0_FOLDER_A, NAME_L0_FOLDER_B, NAME_L0_FOLDER_C };
 | |
|         checkFileList(files, 2, 3, expectedNames);
 | |
|        
 | |
|         // test 2
 | |
|         pagingResults = fileFolderService.list(workingRootNodeRef, true, true, "L1*", null, null, pagingRequest);
 | |
|         
 | |
|         assertNotNull(pagingResults);
 | |
|         assertFalse(pagingResults.hasMoreItems());
 | |
|         assertNull(pagingResults.getTotalResultCount());
 | |
|         
 | |
|         files = pagingResults.getPage();
 | |
|         
 | |
|         // check
 | |
|         expectedNames = new String[]
 | |
|         { };
 | |
|         checkFileList(files, 0, 0, expectedNames);
 | |
| 
 | |
|         // test 3
 | |
|         pagingResults = fileFolderService.list(workingRootNodeRef, true, true, "L0*File*", null, null, pagingRequest);
 | |
|         
 | |
|         assertNotNull(pagingResults);
 | |
|         assertFalse(pagingResults.hasMoreItems());
 | |
|         assertNull(pagingResults.getTotalResultCount());
 | |
|         
 | |
|         files = pagingResults.getPage();
 | |
|         
 | |
|         // check
 | |
|         expectedNames = new String[]
 | |
|         { NAME_L0_FILE_A, NAME_L0_FILE_B };
 | |
|         checkFileList(files, 2, 0, expectedNames);
 | |
| 
 | |
|     }
 | |
| }
 |