mirror of
				https://github.com/Alfresco/alfresco-community-repo.git
				synced 2025-10-22 15:12:38 +00:00 
			
		
		
		
	99981: Merged 5.0.N (5.0.2) to HEAD-BUG-FIX (5.1/Cloud) (PARTIAL MERGE)
      99482: Merged DEV to 5.0.N (5.0.1)
         99198 : MNT-13545: JavaDoc : Inconsistencies between the Java doc and the actual code
            - Changed Javadoc typo in project alfresco-jlan
         99413 : MNT-13545: JavaDoc : Inconsistencies between the Java doc and the actual code
            - Changed Javadoc parameters inconsistence in project alfresco-jlan
         99205 : MNT-13545: JavaDoc : Inconsistencies between the Java doc and the actual code
            - Changed Javadoc parameters inconsistence in project core
         99415 : MNT-13545: JavaDoc : Inconsistencies between the Java doc and the actual code
            - Changed Javadoc parameters inconsistence in project data-model
         99227 : MNT-13545: JavaDoc : Inconsistencies between the Java doc and the actual code
            - Changed Javadoc parameters inconsistence in project file-transfer-receiver
         99416 : MNT-13545: JavaDoc : Inconsistencies between the Java doc and the actual code
            - Changed Javadoc parameters inconsistence in project legacy-lucene
         99417 : MNT-13545: JavaDoc : Inconsistencies between the Java doc and the actual code
            - Changed Javadoc parameters inconsistence in project qa-share
         99418 : MNT-13545: JavaDoc : Inconsistencies between the Java doc and the actual code
            - Changed Javadoc parameters inconsistence in project remote-api
         99427 : MNT-13545: JavaDoc : Inconsistencies between the Java doc and the actual code
            - Changed Javadoc in project Repository, letters S..Z
         99433 : MNT-13545: JavaDoc : Inconsistencies between the Java doc and the actual code
            - Changed Javadoc in project Repository, letters A..R
         99421 : MNT-13545: JavaDoc : Inconsistencies between the Java doc and the actual code
            - Changed Javadoc parameters inconsistence in project share-po
         99247 : MNT-13545: JavaDoc : Inconsistencies between the Java doc and the actual code
            - Changed Javadoc typo in project slingshot
         99248 : MNT-13545: JavaDoc : Inconsistencies between the Java doc and the actual code
            - Changed Javadoc parameters inconsistence in project slingshot
         99424 : MNT-13545: JavaDoc : Inconsistencies between the Java doc and the actual code
            - Changed Javadoc parameters inconsistence in project solr
         99426 : MNT-13545: JavaDoc : Inconsistencies between the Java doc and the actual code
            - Changed Javadoc parameters inconsistence in project solr4
         99253 : MNT-13545: JavaDoc : Inconsistencies between the Java doc and the actual code
            - Changed Javadoc parameters inconsistence in project solr-client
         99259 : MNT-13545: JavaDoc : Inconsistencies between the Java doc and the actual code
            - Changed Javadoc parameters inconsistence in project web-client
         99260 : MNT-13545: JavaDoc : Inconsistencies between the Java doc and the actual code
            - Changed Javadoc parameters inconsistence in project web-framework-commons
git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@100501 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
		
	
		
			
				
	
	
		
			983 lines
		
	
	
		
			32 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
			
		
		
	
	
			983 lines
		
	
	
		
			32 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
| /*
 | |
|  * Copyright (C) 2005-2014 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.imap;
 | |
| 
 | |
| import java.io.IOException;
 | |
| import java.io.Serializable;
 | |
| import java.util.ArrayList;
 | |
| import java.util.Arrays;
 | |
| import java.util.Collection;
 | |
| import java.util.Collections;
 | |
| import java.util.Date;
 | |
| import java.util.LinkedList;
 | |
| import java.util.List;
 | |
| import java.util.Map;
 | |
| import java.util.NavigableMap;
 | |
| 
 | |
| import javax.mail.Flags;
 | |
| import javax.mail.Flags.Flag;
 | |
| import javax.mail.MessagingException;
 | |
| import javax.mail.internet.MimeMessage;
 | |
| import javax.mail.search.SearchTerm;
 | |
| 
 | |
| import org.alfresco.model.ContentModel;
 | |
| import org.alfresco.model.ImapModel;
 | |
| import org.alfresco.repo.imap.AlfrescoImapConst.ImapViewMode;
 | |
| import org.alfresco.repo.imap.ImapService.FolderStatus;
 | |
| 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.FileInfo;
 | |
| import org.alfresco.service.cmr.model.FileNotFoundException;
 | |
| 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.AccessStatus;
 | |
| import org.alfresco.service.namespace.QName;
 | |
| import org.alfresco.util.FileFilterMode;
 | |
| import org.alfresco.util.GUID;
 | |
| import org.alfresco.util.Utf7;
 | |
| import org.apache.commons.logging.Log;
 | |
| import org.apache.commons.logging.LogFactory;
 | |
| 
 | |
| import com.icegreen.greenmail.foedus.util.MsgRangeFilter;
 | |
| import com.icegreen.greenmail.store.FolderException;
 | |
| import com.icegreen.greenmail.store.FolderListener;
 | |
| import com.icegreen.greenmail.store.MailFolder;
 | |
| import com.icegreen.greenmail.store.MessageFlags;
 | |
| import com.icegreen.greenmail.store.SimpleStoredMessage;
 | |
| 
 | |
| /**
 | |
|  * Implementation of greenmail MailFolder. It represents an Alfresco content folder and handles
 | |
|  * appendMessage, copyMessage, expunge (delete), getMessages, getMessage and so requests.
 | |
|  * 
 | |
|  * @author Mike Shavnev
 | |
|  * @author David Ward
 | |
|  */
 | |
| public class AlfrescoImapFolder extends AbstractImapFolder implements Serializable
 | |
| {
 | |
|     private static final long serialVersionUID = -7223111284066976111L;
 | |
| 
 | |
|     /**
 | |
|      * Reference to the {@link FileInfo} object representing the folder.
 | |
|      */
 | |
|     private final FileInfo folderInfo;
 | |
| 
 | |
|     /**
 | |
|      * Name of the folder.
 | |
|      */
 | |
|     private final String folderName;
 | |
| 
 | |
|     private final String folderPath;
 | |
|     
 | |
|     private final String userName;
 | |
| 
 | |
|     private final int mountPointId;
 | |
| 
 | |
|     /**
 | |
|      * Defines view mode.
 | |
|      */
 | |
|     private final ImapViewMode viewMode;
 | |
| 
 | |
|     /**
 | |
|      * Reference to the {@link ImapService} object.
 | |
|      */
 | |
|     private final ImapService imapService;
 | |
|     
 | |
|     /**
 | |
|      * Defines whether the folder is selectable or not.
 | |
|      */
 | |
|     private final boolean selectable;
 | |
| 
 | |
|     private final boolean extractAttachmentsEnabled;
 | |
|     
 | |
|     /**
 | |
|      * Cached Folder status information, validated against a change token.
 | |
|      */
 | |
|     private FolderStatus folderStatus;
 | |
|     
 | |
|     private static final Flags PERMANENT_FLAGS = new Flags();
 | |
| 
 | |
|     static
 | |
|     {
 | |
|         PERMANENT_FLAGS.add(Flags.Flag.ANSWERED);
 | |
|         PERMANENT_FLAGS.add(Flags.Flag.DELETED);
 | |
|         PERMANENT_FLAGS.add(Flags.Flag.DRAFT);
 | |
|         PERMANENT_FLAGS.add(Flags.Flag.FLAGGED);
 | |
|         PERMANENT_FLAGS.add(Flags.Flag.SEEN);
 | |
|     }
 | |
|     
 | |
|     public boolean isExtractAttachmentsEnabled() 
 | |
|     {
 | |
|         return extractAttachmentsEnabled;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Protected constructor for the hierarchy delimiter
 | |
|      */
 | |
|     AlfrescoImapFolder(String userName, ImapService imapService, ServiceRegistry serviceRegistry)
 | |
|     {
 | |
|         this(null, userName, "", "", null, imapService, serviceRegistry, false, false, 0);
 | |
|     }
 | |
|         
 | |
| 
 | |
|     /**
 | |
|      * Constructs {@link AlfrescoImapFolder} object.
 | |
|      * 
 | |
|      * @param folderInfo - reference to the {@link FileInfo} object representing the folder.
 | |
|      * @param userName - name of user (e.g. "admin" for admin user).
 | |
|      * @param folderName - name of the folder.
 | |
|      * @param folderPath - path of the folder.
 | |
|      * @param viewMode - defines view mode. Can be one of the following: {@link ImapViewMode#ARCHIVE} or {@link ImapViewMode#VIRTUAL}.
 | |
|      * @param extractAttachmentsEnabled
 | |
|      * @param imapService
 | |
|      * @param serviceRegistry
 | |
|      * @param mountPointId - id of the mount point.
 | |
|      */
 | |
|     public AlfrescoImapFolder(
 | |
|             FileInfo folderInfo,
 | |
|             String userName,
 | |
|             String folderName,
 | |
|             String folderPath,
 | |
|             ImapViewMode viewMode,
 | |
|             boolean extractAttachmentsEnabled,
 | |
|             ImapService imapService,
 | |
|             ServiceRegistry serviceRegistry,
 | |
|             int mountPointId)
 | |
|     {
 | |
|         this(folderInfo, userName, folderName, folderPath, viewMode, imapService, serviceRegistry, null, extractAttachmentsEnabled, mountPointId);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Constructs {@link AlfrescoImapFolder} object.
 | |
|      * 
 | |
|      * @param folderInfo - reference to the {@link FileInfo} object representing the folder.
 | |
|      * @param userName - name of the user (e.g. "admin" for admin user).
 | |
|      * @param folderName - name of the folder.
 | |
|      * @param folderPath - path of the folder.
 | |
|      * @param viewMode - defines view mode. Can be one of the following: {@link ImapViewMode#ARCHIVE} or {@link ImapViewMode#VIRTUAL}.
 | |
|      * @param imapService - the IMAP service.
 | |
|      * @param serviceRegistry
 | |
|      * @param selectable - defines whether the folder is selectable or not.
 | |
|      * @param extractAttachmentsEnabled
 | |
|      * @param mountPointId
 | |
|      */
 | |
|     public AlfrescoImapFolder(
 | |
|             FileInfo folderInfo,
 | |
|             String userName,
 | |
|             String folderName,
 | |
|             String folderPath,
 | |
|             ImapViewMode viewMode,
 | |
|             ImapService imapService,
 | |
|             ServiceRegistry serviceRegistry,
 | |
|             Boolean selectable,
 | |
|             boolean extractAttachmentsEnabled,
 | |
|             int mountPointId)
 | |
|     {
 | |
|         super(serviceRegistry);
 | |
|         this.folderInfo = folderInfo;
 | |
|         this.userName = userName;
 | |
|         this.folderName = folderName != null ? folderName : (folderInfo != null ? folderInfo.getName() : null);
 | |
|         this.folderPath = folderPath;
 | |
|         this.viewMode = viewMode != null ? viewMode : ImapViewMode.ARCHIVE;
 | |
|         this.extractAttachmentsEnabled = extractAttachmentsEnabled;
 | |
|         this.imapService = imapService;
 | |
|         
 | |
|         // MailFolder object can be null if it is used to obtain hierarchy delimiter by LIST command:
 | |
|         // Example:
 | |
|         // C: 2 list "" ""
 | |
|         // S: * LIST () "." ""
 | |
|         // S: 2 OK LIST completed.
 | |
|         if (folderInfo != null)
 | |
|         {
 | |
|             if (selectable == null)
 | |
|             {
 | |
|                 // isSelectable();
 | |
|                 Boolean storedSelectable = !serviceRegistry.getNodeService().hasAspect(folderInfo.getNodeRef(), ImapModel.ASPECT_IMAP_FOLDER_NONSELECTABLE);
 | |
|                 if (storedSelectable == null)
 | |
|                 {
 | |
|                     this.selectable = true;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     this.selectable = storedSelectable;
 | |
|                 }
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 this.selectable = selectable;
 | |
|             }
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             this.selectable = false;
 | |
|         }
 | |
|         
 | |
|         this.mountPointId = mountPointId;
 | |
|     }
 | |
|     
 | |
|     /*
 | |
|      * (non-Javadoc)
 | |
|      * @see com.icegreen.greenmail.store.MailFolder#getFullName()
 | |
|      */
 | |
|     @Override
 | |
|     public String getFullName()
 | |
|     {
 | |
|         return Utf7.encode(AlfrescoImapConst.USER_NAMESPACE + AlfrescoImapConst.HIERARCHY_DELIMITER + this.userName
 | |
|                 + AlfrescoImapConst.HIERARCHY_DELIMITER + getFolderPath(), Utf7.UTF7_MODIFIED);
 | |
|     }
 | |
| 
 | |
|     /* (non-Javadoc)
 | |
|      * @see com.icegreen.greenmail.store.MailFolder#getName()
 | |
|      */
 | |
|     @Override
 | |
|     public String getName()
 | |
|     {
 | |
|         return this.folderName;
 | |
|     }
 | |
| 
 | |
|     /* (non-Javadoc)
 | |
|      * @see com.icegreen.greenmail.store.MailFolder#isSelectable()
 | |
|      */
 | |
|     @Override
 | |
|     public boolean isSelectable()
 | |
|     {
 | |
|         return this.selectable;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns the contents of this folder.
 | |
|      * 
 | |
|      * @return A sorted map of UIDs to FileInfo objects.
 | |
|      */
 | |
|     private NavigableMap<Long, FileInfo> searchMails()
 | |
|     {
 | |
|         return getFolderStatus().search;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Invalidates the current cached state
 | |
|      * 
 | |
|      * @return <code>true</code> if this instance is still valid for reuse
 | |
|      */
 | |
|     public boolean reset()
 | |
|     {
 | |
|         this.folderStatus = null;
 | |
|         return new CommandCallback<Boolean>()
 | |
|         {
 | |
|             public Boolean command() throws Throwable
 | |
|             {
 | |
|                 return serviceRegistry.getNodeService().exists(folderInfo.getNodeRef());
 | |
|             }
 | |
|         }.run(true);
 | |
|     }
 | |
| 
 | |
|     protected FolderStatus getFolderStatus()
 | |
|     {
 | |
|         CommandCallback<FolderStatus> command = new CommandCallback<FolderStatus>()
 | |
|         {
 | |
|             public FolderStatus command() throws Throwable
 | |
|             {
 | |
|                 return imapService.getFolderStatus(userName, folderInfo.getNodeRef(), viewMode);
 | |
|             }
 | |
|         };
 | |
|         return this.folderStatus = command.run();
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Appends message to the folder.
 | |
|      * 
 | |
|      * @param message - message.
 | |
|      * @param flags - message flags.
 | |
|      * @param internalDate - not used. Current date used instead.
 | |
|      */
 | |
|     @Override
 | |
|     protected long appendMessageInternal(
 | |
|             MimeMessage message,
 | |
|             Flags flags,
 | |
|             Date internalDate)
 | |
|             throws FileExistsException, FileNotFoundException, IOException, MessagingException 
 | |
|     {
 | |
|         long uid;
 | |
|         NodeRef sourceNodeRef = extractNodeRef(message);
 | |
|         if (sourceNodeRef != null)
 | |
|         {
 | |
|             uid = copyOrMoveNode(this.folderInfo, message, flags, sourceNodeRef, false);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             uid = createMimeMessageInFolder(this.folderInfo, message, flags);
 | |
|         }
 | |
|         // Invalidate current folder status
 | |
|         this.folderStatus = null;
 | |
|         return uid;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Moves the node <code>sourceNodeRef</code> extracted from the message id.
 | |
|      * A part of a complex move operation.
 | |
|      * 
 | |
|      * @param folderInfo
 | |
|      * @param message
 | |
|      * @param flags
 | |
|      * @param sourceNodeRef
 | |
|      * @return UUID of the moved node
 | |
|      * @throws FileExistsException
 | |
|      * @throws FileNotFoundException
 | |
|      */
 | |
|     @SuppressWarnings("deprecation")
 | |
|     private long copyOrMoveNode(FileInfo folderInfo, MimeMessage message, Flags flags, NodeRef sourceNodeRef, boolean move)
 | |
|             throws FileExistsException, FileNotFoundException
 | |
|     {
 | |
|         FileFolderService fileFolderService = serviceRegistry.getFileFolderService();
 | |
|         FileFilterMode.setClient(FileFilterMode.Client.imap);
 | |
|         FileInfo messageFile = null;
 | |
|         if (move)
 | |
|         {
 | |
|             fileFolderService.setHidden(sourceNodeRef, false);
 | |
|             messageFile = fileFolderService.move(sourceNodeRef, folderInfo.getNodeRef(), null);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             NodeRef newNodeRef = serviceRegistry.getCopyService().copyAndRename(sourceNodeRef, folderInfo.getNodeRef(), ContentModel.ASSOC_CONTAINS, null, false);
 | |
|             fileFolderService.setHidden(newNodeRef, false);
 | |
|             messageFile = fileFolderService.getFileInfo(newNodeRef);
 | |
|         }
 | |
|         final long newMessageUid = (Long) messageFile.getProperties().get(ContentModel.PROP_NODE_DBID);
 | |
|         
 | |
|         imapService.persistMessageHeaders(messageFile.getNodeRef(), message);
 | |
|         
 | |
|         Flags newFlags = new Flags(flags);
 | |
|         newFlags.add(Flag.RECENT);
 | |
|         
 | |
|         imapService.setFlags(messageFile, newFlags, true);
 | |
|         imapService.setFlag(messageFile, Flag.DELETED, false);
 | |
|         
 | |
|         return newMessageUid;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Extract a <code>NodeRef</code> from the message id.
 | |
|      * <br>Typical message id is "<74bad8aa-75a5-4063-8e46-9d1b5737f43b@alfresco.org>"
 | |
|      * <br>See {@link AbstractMimeMessage#updateMessageID()}
 | |
|      * 
 | |
|      * @param message
 | |
|      * @return null if nothing is found
 | |
|      */
 | |
|     private NodeRef extractNodeRef(MimeMessage message)
 | |
|     {
 | |
|         String uuid = null;
 | |
|         String messageId = null;
 | |
|         NodeRef result = null;
 | |
|         NodeService nodeService = serviceRegistry.getNodeService();
 | |
|         try
 | |
|         {
 | |
|             messageId = message.getMessageID();
 | |
|         }
 | |
|         catch (MessagingException me)
 | |
|         {
 | |
|             // we cannot use message id to extract nodeRef
 | |
|         }
 | |
|         
 | |
|         if (messageId != null)
 | |
|         {
 | |
|             if (messageId.startsWith("<"))
 | |
|             {
 | |
|                 messageId = messageId.substring(1);
 | |
|             }
 | |
|             if (messageId.indexOf("@") != -1)
 | |
|             {
 | |
|                 uuid = messageId.substring(0, messageId.indexOf("@"));
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 uuid = messageId;
 | |
|             }
 | |
|             result = new NodeRef(StoreRef.PROTOCOL_WORKSPACE, "SpacesStore", uuid);
 | |
|             if (nodeService.exists(result) == false)
 | |
|             {
 | |
|                 result = null;
 | |
|             }
 | |
|         }
 | |
|         
 | |
|         if(result == null)
 | |
|         {
 | |
|             //check X-Alfresco-NodeRef-ID header
 | |
|             try
 | |
|             {
 | |
|                 if (message.getHeader(AlfrescoImapConst.X_ALF_NODEREF_ID) != null)
 | |
|                 {
 | |
|                     uuid = message.getHeader(AlfrescoImapConst.X_ALF_NODEREF_ID)[0];
 | |
|                     result = new NodeRef(StoreRef.PROTOCOL_WORKSPACE, "SpacesStore", uuid);
 | |
|                     if (nodeService.exists(result) == false)
 | |
|                     {
 | |
|                         result = null;
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             catch (MessagingException e)
 | |
|             {
 | |
|                 //Do nothing
 | |
|             }
 | |
|         }
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Determine if it is a complex move operation, which consists of a create superseded by a delete.
 | |
|      * 
 | |
|      * @param sourceNodeRef
 | |
|      * @return 
 | |
|      */
 | |
|     @SuppressWarnings("deprecation")
 | |
|     private boolean isMoveOperation(NodeRef sourceNodeRef)
 | |
|     {
 | |
|         if (sourceNodeRef != null)
 | |
|         {
 | |
|             NodeService nodeService = serviceRegistry.getNodeService();
 | |
|             if (nodeService.exists(sourceNodeRef))
 | |
|             {
 | |
|                 FileFolderService fileFolderService = serviceRegistry.getFileFolderService();
 | |
|                 FileInfo node = fileFolderService.getFileInfo(sourceNodeRef);
 | |
|                 if (node != null)
 | |
|                 {
 | |
|                     if (fileFolderService.isHidden(sourceNodeRef))
 | |
|                     {
 | |
|                         return true;
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Copies message with the given UID to the specified {@link MailFolder}.
 | |
|      * 
 | |
|      * @param uid - UID of the message
 | |
|      * @param toFolder - reference to the destination folder.
 | |
|      * @throws MessagingException 
 | |
|      * @throws IOException 
 | |
|      * @throws FileNotFoundException 
 | |
|      * @throws FileExistsException 
 | |
|      */
 | |
|     @Override
 | |
|     protected long copyMessageInternal(
 | |
|             long uid, MailFolder toFolder)
 | |
|             throws MessagingException, FileExistsException, FileNotFoundException, IOException 
 | |
|     {
 | |
|         AlfrescoImapFolder toImapMailFolder = (AlfrescoImapFolder) toFolder;
 | |
| 
 | |
|         NodeRef destFolderNodeRef = toImapMailFolder.getFolderInfo().getNodeRef();
 | |
| 
 | |
|         FileInfo sourceMessageFileInfo = searchMails().get(uid);
 | |
| 
 | |
|         if (serviceRegistry.getNodeService().hasAspect(sourceMessageFileInfo.getNodeRef(), ImapModel.ASPECT_IMAP_CONTENT))
 | |
|         {
 | |
|                 //Generate body of message
 | |
|             MimeMessage newMessage = new ImapModelMessage(sourceMessageFileInfo, serviceRegistry, true);
 | |
|             return toImapMailFolder.appendMessageInternal(newMessage, imapService.getFlags(sourceMessageFileInfo), new Date());
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             String fileName = (String) serviceRegistry.getNodeService().getProperty(sourceMessageFileInfo.getNodeRef(), ContentModel.PROP_NAME);
 | |
|             String newFileName = imapService.generateUniqueFilename(destFolderNodeRef, fileName);
 | |
|             FileInfo messageFileInfo = serviceRegistry.getFileFolderService().copy(sourceMessageFileInfo.getNodeRef(), destFolderNodeRef, newFileName);
 | |
|             return (Long)messageFileInfo.getProperties().get(ContentModel.PROP_NODE_DBID);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Marks all messages in the folder as deleted using {@link Flags.Flag#DELETED} flag.
 | |
|      */
 | |
|     @Override
 | |
|     public void deleteAllMessagesInternal() throws FolderException
 | |
|     {
 | |
|         if (isReadOnly())
 | |
|         {
 | |
|             throw new FolderException("Can't delete all - Permission denied");
 | |
|         }
 | |
|         
 | |
|         for (Map.Entry<Long, FileInfo> entry : searchMails().entrySet())
 | |
|         {
 | |
|             imapService.setFlag(entry.getValue(), Flags.Flag.DELETED, true);
 | |
|             // comment out to physically remove content.
 | |
|             // fileFolderService.delete(fileInfo.getNodeRef());
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Deletes messages marked with {@link Flags.Flag#DELETED}. Note that this message deletes all messages with this flag.
 | |
|      */
 | |
|     @Override
 | |
|     protected void expungeInternal() throws FolderException
 | |
|     {
 | |
|         if (isReadOnly())
 | |
|         {
 | |
|             throw new FolderException("Can't expunge - Permission denied");
 | |
|         }
 | |
| 
 | |
|         for (Map.Entry<Long, FileInfo> entry : searchMails().entrySet())
 | |
|         {
 | |
|             imapService.expungeMessage(entry.getValue());
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Deletes messages marked with {@link Flags.Flag#DELETED}. Note that this message deletes the message with current uid
 | |
|      */
 | |
|     @Override
 | |
|     protected void expungeInternal(long uid) throws Exception
 | |
|     {
 | |
|         if (isReadOnly())
 | |
|         {
 | |
|             throw new FolderException("Can't expunge - Permission denied");
 | |
|         }
 | |
|         
 | |
|         FileInfo messageFileInfo = searchMails().get(uid);
 | |
| 
 | |
|         imapService.expungeMessage(messageFileInfo);
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Returns the MSN number of the first unseen message.
 | |
|      * 
 | |
|      * @return MSN number of the first unseen message.
 | |
|      */
 | |
|     @Override
 | |
|     public int getFirstUnseen()
 | |
|     {
 | |
|         return getFolderStatus().firstUnseen;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns message by its UID.
 | |
|      * 
 | |
|      * @param uid - UID of the message.
 | |
|      * @return message.
 | |
|      * @throws MessagingException 
 | |
|      */
 | |
|     @Override
 | |
|     protected SimpleStoredMessage getMessageInternal(long uid) throws MessagingException
 | |
|     {
 | |
|         if (logger.isDebugEnabled())
 | |
|         {
 | |
|             logger.debug("[getMessageInternal] " + this);
 | |
|         }
 | |
|         FileInfo mesInfo = searchMails().get(uid); 
 | |
|         if (mesInfo == null)
 | |
|         {
 | |
|             return null;
 | |
|         }
 | |
|         return imapService.getMessage(mesInfo);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns count of the messages in the folder.
 | |
|      * 
 | |
|      * @return Count of the messages.
 | |
|      */
 | |
|     @Override
 | |
|     public int getMessageCount()
 | |
|     {
 | |
|         return getFolderStatus().messageCount;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns UIDs of all messages in the folder.
 | |
|      * 
 | |
|      * @return UIDS of the messages.
 | |
|      */
 | |
|     @Override
 | |
|     public long[] getMessageUids()
 | |
|     {
 | |
|         if (logger.isDebugEnabled())
 | |
|         {
 | |
|             logger.debug("[getMessageUidsInternal] " + this);
 | |
|         }
 | |
|         
 | |
|         Collection<Long> uidSet = searchMails().keySet();
 | |
|         long[] uids = new long[uidSet.size()];
 | |
|         int i = 0;
 | |
|         for (Long uid : uidSet)
 | |
|         {
 | |
|             uids[i++] = uid;
 | |
|         }
 | |
|         return uids;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns list of all messages in the folder.
 | |
|      * 
 | |
|      * @return list of {@link SimpleStoredMessage} objects.
 | |
|      */
 | |
|     @Override
 | |
|     protected List<SimpleStoredMessage> getMessagesInternal()
 | |
|     {
 | |
|         if (logger.isDebugEnabled())
 | |
|         {
 | |
|             logger.debug("[getMessagesInternal] " + this);
 | |
|         }
 | |
|         return convertToMessages(searchMails().values());
 | |
|     }
 | |
| 
 | |
|     private List<SimpleStoredMessage> convertToMessages(Collection<FileInfo> fileInfos)
 | |
|     {
 | |
|         if (logger.isDebugEnabled())
 | |
|         {
 | |
|             logger.debug("[convertToMessages] " + this);
 | |
|         }
 | |
|         if (fileInfos == null || fileInfos.size() == 0)
 | |
|         {
 | |
|             logger.debug("[convertToMessages] - fileInfos is empty or null");
 | |
|             return Collections.emptyList();
 | |
|         }
 | |
|         List<SimpleStoredMessage> result = new LinkedList<SimpleStoredMessage>();
 | |
|         for (FileInfo fileInfo : fileInfos)
 | |
|         {
 | |
|             try
 | |
|             {
 | |
|                 result.add(imapService.createImapMessage(fileInfo, true));
 | |
|                 if (logger.isDebugEnabled())
 | |
|                 {
 | |
|                     logger.debug("[convertToMessages] Message added: " + fileInfo.getName());
 | |
|                 }
 | |
|             }
 | |
|             catch (MessagingException e)
 | |
|             {
 | |
|                 logger.warn("[convertToMessages] Invalid message! File name:" + fileInfo.getName(), e);
 | |
|             }
 | |
|         }
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns list of messages by filter.
 | |
|      * 
 | |
|      * @param msgRangeFilter - {@link MsgRangeFilter} object representing filter.
 | |
|      * @return list of filtered messages.
 | |
|      */
 | |
|     @Override
 | |
|     protected List<SimpleStoredMessage> getMessagesInternal(MsgRangeFilter msgRangeFilter)
 | |
|     {
 | |
|         throw new UnsupportedOperationException("IMAP implementation doesn't support POP3 requests");
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns message sequence number in the folder by its UID.
 | |
|      * 
 | |
|      * @param uid - message UID.
 | |
|      * @return message sequence number.
 | |
|      * @throws FolderException if no message with given UID.
 | |
|      */
 | |
|     @Override
 | |
|     public int getMsn(long uid) throws FolderException
 | |
|     {
 | |
|         NavigableMap<Long, FileInfo> messages = searchMails();
 | |
|         if (!messages.containsKey(uid))
 | |
|         {
 | |
|             throw new FolderException("No such message.");            
 | |
|         }
 | |
|         return messages.headMap(uid, true).size();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns the list of messages that have no {@link Flags.Flag#DELETED} flag set for current user.
 | |
|      * 
 | |
|      * @return the list of non-deleted messages.
 | |
|      */
 | |
|     @Override
 | |
|     protected List<SimpleStoredMessage> getNonDeletedMessagesInternal()
 | |
|     {
 | |
|         if (logger.isDebugEnabled())
 | |
|         {
 | |
|             logger.debug("[getNonDeletedMessagesInternal] " + this);
 | |
|         }
 | |
|         List<SimpleStoredMessage> result = new ArrayList<SimpleStoredMessage>();
 | |
| 
 | |
|         Collection<SimpleStoredMessage> values = getMessagesInternal();
 | |
|         for (SimpleStoredMessage message : values)
 | |
|         {
 | |
|             if (!getFlags(message).contains(Flags.Flag.DELETED))
 | |
|             {
 | |
|                 result.add(message);
 | |
|             }
 | |
| 
 | |
|         }
 | |
|         if (logger.isDebugEnabled() && folderInfo != null)
 | |
|         {
 | |
|             logger.debug(folderInfo.getName() + " - Non deleted messages count:" + result.size());
 | |
|         }
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns permanent flags.
 | |
|      * 
 | |
|      * @return {@link Flags} object containing flags.
 | |
|      */
 | |
|     @Override
 | |
|     public Flags getPermanentFlags()
 | |
|     {
 | |
|         return PERMANENT_FLAGS;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns count of messages with {@link Flags.Flag#RECENT} flag.
 | |
|      * If {@code reset} parameter is {@code true} - removes {@link Flags.Flag#RECENT} flag from
 | |
|      * the message for current user.
 | |
|      * 
 | |
|      * @param reset - if true the {@link Flags.Flag#RECENT} will be deleted for current user if exists.
 | |
|      * @return returns count of recent messages.
 | |
|      */
 | |
|     @Override
 | |
|     public int getRecentCount(boolean reset)
 | |
|     {
 | |
|         int recent = getFolderStatus().recentCount;
 | |
|         if (reset && recent > 0)
 | |
|         {
 | |
|             CommandCallback<Void> command = new CommandCallback<Void>()
 | |
|             {
 | |
|                 public Void command() throws Throwable
 | |
|                 {
 | |
|                     for (FileInfo fileInfo : folderStatus.search.values())
 | |
|                     {
 | |
|                         Flags flags = imapService.getFlags(fileInfo);
 | |
|                         if (flags.contains(Flags.Flag.RECENT))
 | |
|                         {
 | |
|                             imapService.setFlag(fileInfo, Flags.Flag.RECENT, false);
 | |
|                         }
 | |
|                     }
 | |
|                     return null;
 | |
|                 }
 | |
|             };
 | |
|             try
 | |
|             {
 | |
|                 command.run();
 | |
|             }
 | |
|             catch (AccessDeniedException ade)
 | |
|             {
 | |
|                 if (logger.isDebugEnabled())
 | |
|                 {
 | |
|                     logger.debug("Access denied to reset RECENT FLAG");
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         return recent;        
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns UIDNEXT value of the folder.
 | |
|      * 
 | |
|      * @return UIDNEXT value.
 | |
|      */
 | |
|     @Override
 | |
|     public long getUidNext()
 | |
|     {
 | |
|         NavigableMap<Long, FileInfo> search = getFolderStatus().search; 
 | |
|         return search.isEmpty() ? 1 : search.lastKey() + 1;
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Returns UIDVALIDITY value of the folder.
 | |
|      * 
 | |
|      * @return UIDVALIDITY value.
 | |
|      */
 | |
|     @Override
 | |
|     public long getUidValidity()
 | |
|     {
 | |
|         return getFolderStatus().uidValidity / 1000L + mountPointId;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns count of the messages with {@link Flags.Flag#SEEN} in the folder for the current user.
 | |
|      * 
 | |
|      * @return Count of the unseen messages for current user.
 | |
|      */
 | |
|     @Override
 | |
|     public int getUnseenCount()
 | |
|     {
 | |
|         return getFolderStatus().unseenCount;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Replaces flags for the message with the given UID. If {@code addUid} is set to {@code true}
 | |
|      * {@link FolderListener} objects defined for this folder will be notified.
 | |
|      * {@code silentListener} can be provided - this listener wouldn't be notified.
 | |
|      * 
 | |
|      * @param flags - new flags.
 | |
|      * @param uid - message UID.
 | |
|      * @param silentListener - listener that shouldn't be notified.
 | |
|      * @param addUid - defines whether or not listeners be notified.
 | |
|      * @throws FolderException 
 | |
|      * @throws MessagingException 
 | |
|      */
 | |
|     @Override
 | |
|     protected void replaceFlagsInternal(
 | |
|             Flags flags,
 | |
|             long uid,
 | |
|             FolderListener silentListener,
 | |
|             boolean addUid)
 | |
|             throws FolderException, MessagingException 
 | |
|     {
 | |
|         int msn = getMsn(uid);
 | |
|         FileInfo fileInfo = searchMails().get(uid);
 | |
|         imapService.setFlags(fileInfo, MessageFlags.ALL_FLAGS, false);
 | |
|         imapService.setFlags(fileInfo, flags, true);
 | |
|         
 | |
|         Long uidNotification = addUid ? uid : null;
 | |
|         notifyFlagUpdate(msn, flags, uidNotification, silentListener);
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     protected long[] searchInternal(SearchTerm searchTerm)
 | |
|     {
 | |
|         List<SimpleStoredMessage> messages = getMessages();
 | |
|         long[] result = new long[messages.size()];
 | |
|         int i = 0;
 | |
|         
 | |
|         for (SimpleStoredMessage message : messages)
 | |
|         {
 | |
|             if (searchTerm.match(message.getMimeMessage()))
 | |
|             {
 | |
|                 result[i] = message.getUid();
 | |
|                 i++;
 | |
|             }
 | |
|         }
 | |
|         return Arrays.copyOfRange(result, 0, i);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Sets flags for the message with the given UID. If {@code addUid} is set to {@code true}
 | |
|      * {@link FolderListener} objects defined for this folder will be notified.
 | |
|      * {@code silentListener} can be provided - this listener wouldn't be notified.
 | |
|      * 
 | |
|      * @param flags - new flags.
 | |
|      * @param value - flags value.
 | |
|      * @param uid - message UID.
 | |
|      * @param silentListener - listener that shouldn't be notified.
 | |
|      * @param addUid - defines whether or not listeners be notified.
 | |
|      * @throws MessagingException 
 | |
|      * @throws FolderException 
 | |
|      */
 | |
|     @Override
 | |
|     protected void setFlagsInternal(
 | |
|             Flags flags,
 | |
|             boolean value,
 | |
|             long uid,
 | |
|             FolderListener silentListener,
 | |
|             boolean addUid)
 | |
|             throws MessagingException, FolderException 
 | |
|     {
 | |
|         int msn = getMsn(uid);
 | |
|         FileInfo fileInfo = searchMails().get(uid);
 | |
|         imapService.setFlags(fileInfo, flags, value);
 | |
|         
 | |
|         Long uidNotification = null;
 | |
|         if (addUid)
 | |
|         {
 | |
|             uidNotification = new Long(uid);
 | |
|         }
 | |
|         notifyFlagUpdate(msn, flags, uidNotification, silentListener);
 | |
| 
 | |
|     }
 | |
| 
 | |
|     private Flags getFlags(SimpleStoredMessage mess)
 | |
|     {
 | |
|         return ((AbstractMimeMessage) mess.getMimeMessage()).getFlags();
 | |
|     }
 | |
| 
 | |
|     // ----------------------Getters and Setters----------------------------
 | |
| 
 | |
|     public String getFolderPath()
 | |
|     {
 | |
|         return this.folderPath;
 | |
|     }
 | |
| 
 | |
|     public FileInfo getFolderInfo()
 | |
|     {
 | |
|         return folderInfo;
 | |
|     }
 | |
|     
 | |
|     /* (non-Javadoc)
 | |
|      * @see org.alfresco.repo.imap.AbstractImapFolder#isMarkedInternal()
 | |
|      */
 | |
|     @Override
 | |
|     public boolean isMarked()
 | |
|     {
 | |
|         FolderStatus folderStatus = getFolderStatus();
 | |
|         return folderStatus.recentCount > 0 || folderStatus.unseenCount > 0;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Whether the folder is read-only for user.
 | |
|      * 
 | |
|      * @return {@code boolean}
 | |
|      */
 | |
|     @Override
 | |
|     protected boolean isReadOnly()
 | |
|     {
 | |
|         AccessStatus status = serviceRegistry.getPublicServiceAccessService().hasAccess(ServiceRegistry.NODE_SERVICE.getLocalName(), "createNode", folderInfo.getNodeRef(), null, null, null);
 | |
|         //serviceRegistry.getPermissionService().hasPermission(folderInfo.getNodeRef(), PermissionService.WRITE);
 | |
|         return  status == AccessStatus.DENIED;
 | |
|     }
 | |
| 
 | |
|     public ImapViewMode getViewMode()
 | |
|     {
 | |
|         return viewMode;
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      *  Creates the EML message in the specified folder.
 | |
|      *  
 | |
|      *  @param folderFileInfo The folder to create message in.
 | |
|      *  @param message The original MimeMessage.
 | |
|      *  @return ID of the new message created 
 | |
|      * @throws FileNotFoundException 
 | |
|      * @throws FileExistsException 
 | |
|      * @throws MessagingException 
 | |
|      * @throws IOException 
 | |
|      */
 | |
|     private long createMimeMessageInFolder(
 | |
|             FileInfo folderFileInfo,
 | |
|             MimeMessage message,
 | |
|             Flags flags)
 | |
|             throws FileExistsException, FileNotFoundException, IOException, MessagingException 
 | |
|     {
 | |
|         String name = AlfrescoImapConst.MESSAGE_PREFIX + GUID.generate();
 | |
|         FileFolderService fileFolderService = serviceRegistry.getFileFolderService();
 | |
|         FileInfo messageFile = fileFolderService.create(folderFileInfo.getNodeRef(), name, ContentModel.TYPE_CONTENT);
 | |
|         final long newMessageUid = (Long) messageFile.getProperties().get(ContentModel.PROP_NODE_DBID);
 | |
|         name = AlfrescoImapConst.MESSAGE_PREFIX  + newMessageUid + AlfrescoImapConst.EML_EXTENSION;
 | |
|         fileFolderService.rename(messageFile.getNodeRef(), name);
 | |
|         Flags newFlags = new Flags(flags);
 | |
|         newFlags.add(Flag.RECENT);
 | |
|         imapService.setFlags(messageFile, newFlags, true);
 | |
|         
 | |
|         if (extractAttachmentsEnabled)
 | |
|         {
 | |
|             imapService.extractAttachments(messageFile.getNodeRef(), message);
 | |
|         }
 | |
|         // Force persistence of the message to the repository
 | |
|         new IncomingImapMessage(messageFile, serviceRegistry, message);
 | |
|         return newMessageUid;        
 | |
|     }
 | |
| }
 |