mirror of
https://github.com/Alfresco/alfresco-community-repo.git
synced 2025-06-30 18:15:39 +00:00
30947: ALF-10619: Merged PATCHES/V3.1.2 to V3.4-BUG-FIX 30884: ALF-10588: Another possible race condition resulting in out of sync transactions - found on SQL Server and JBoss in 3.1.2 - FTS could process updated and deleted nodes in the same transaction before the tracker got to them, leaving behind the correct transaction ID and deleted nodes but undeleted container docs! - We now have to validate all deletions have been honoured when index tracking 30890: ALF-10588: Temporarily disable FTS during IndexCheckServiceImplTest - Otherwise can get confused by intermediate FTS state of its own nodes! 30894: ALF-10588: Correction to deletion checking - Only search for deleted nodes, not updated ones too! 30948: ALF-10619: Fixed merge issue 30982: - ALF-10503 60k Site Performance: Admin Console | Groups: search with a value that matches all 60 groups: maxClasuesCount=10000 - ALF-10511 60k Site Performance: Admin Console | Users | Edit User | Group Search with a value that matches all 60 groups: maxClauseCount=10000 - ALF-10608 60k Site Performance: Searching for a group to add to a site with a value that matches all 60 groups: maxClauseCount=10000 - ALF-10515 60k Site Performance: Edit Group Display Name: The first time, nothing appears to happen for 10 seconds after pressing [Save] - ALF-10514 60k Site Performance: Admin Console | Groups | Search | Delete Group: no feedback to user for 20 seconds after clicking delete icon 30985: Increases in node, property and aspect caches. 30987: Merged DEV/TEMPORARY to V3.4-BUG-FIX 30984: ALF-9880 : ContentGet web script throws NullPointerException for nodes missing cm:modified property The check for null was added for cm:modified property (similar to BaseDownloadContentServlet). 30995: Fix for ALF-9021 30996: ALF-10324 Cannot disable Home Folder Creation - Bug introduced into V3.1 on the 8 March 2010 - ChainingUserRegistrySynchronizerTest enhanced to check for this - Fix to PersonService: Home folder was not being created for 'missing' persons - PersonService: Changed autoCreate parameters to more descriptive names (okay long) and updated Javadoc 30998: ALF-10512 60k Site Performance: Clicking on Sites (left hand side) in the Repository browser causes transactional limit to be reached - Changed node, aspect, property and parentAssoc cache sizes (based on Derek's Skype message) 31006: ALF-10512 60k Site Performance: Clicking on Sites (left hand side) in the Repository browser causes a transactional limit to be reached - Having changed cache sizes in previous commit, the nodeOwner and acl transactional caches were then blown with test case for ALF-10512 Changed to 20k from 10k. Tried 15k but it still had a problem. 31052: Fix for ALF-10520 Merged HEAD to V3.4-BUG-FIX 31051: Performance improvements for Share Repository browser queries. DB with ~50,000 nodes under Company Home: Before: - I'm Editing - 16 secs, Favorites - 17 secs, Tag - 14 secs After: - I'm Editing - 1.5 secs, Favorites - 1.2 secs, Tag - 1.25 secs 31058: ALF-10324 Cannot disable Home Folder Creation - ChainingUserRegistrySynchronizerTest check using personService with both eager and non eager home folder creation 31064: ALF-9360: Merged PATCHES/V3.4.4 to V3.4-BUG-FIX 30244: Merged DEV/DAVEW/IMAP_NEW to PATCHES/V3.4.4 29635: Rework of IMAP to use lightweight caching and correctly set UIDVALIDITY, NEXTUID and Marked / Unmarked state 29668: 1. Changed get AlfrescoImapFolder.getFullNameInternal to be dynamic for cache support 29692: 1. Reverts changed in AlfrescoImapServer to allow ImapHostManager to be a session key for folder. 2. getFlags relies on FileInfo.getProperties() 29741: 1. Changed AbstractMimeMessage.updateMessageId() to follow RFC2822 (3.6.4. Identification fields) 2. Changed ImapServiceImpl to handle absent folders and return "NO" reply to a client. 3. Changed ImapServiceImpl that behaviours don't fail when Alfresco is being first time bootstrapped with IMAP enabled. 4. Cleared AlfrescoImapFolder constructor. 5. Fixed SelectCommand's response to adhere RFC3501 (6.3.1. SELECT Command) 6. Fixed CommandParser to be able parse the flag which is not surrounded by braces (STORE 2:4 +FLAGS \Deleted) 30235: Completion of IMAP rework - Scalable caching - Proper transactional cache for assembled messages - No more assumption that EHcache will always hold entire folder set at once (and perhaps it can't) - Per session (TCP connection) cache of accessed folders - Session cache validation via a 'change token' that is incremented on all significant events - Folder status attributes evaluated once and reused until the change token changes - Now only changed folders need to be queried on an IMAP sync and the server doesn't have to hold all folders in memory - User's view is consistent with their security permissions - Simplification / overhaul of ImapServiceImpl including efficient recursive path building and matching - AlfrescoImapFolder immutable as it should be - Greenmail fixes - Fixed quoting of mailbox names - Fixed hanging problem in ImapRequestLineReader - regression caused by our 8 bit encoding fix. Avoid using an InputStreamReader to read ISO-8859-1 bytes as it has an internal buffer. 30275: Fix failing IMAP tests broken by my recent refactor! - Fixed greenmail conversion of ISO-8859-1 bytes to chars - Transaction read write attributes on service - Read only commands on AbstractImapFolder - Imap aspect properties must be managed as SYSTEM user - Restored persistence of new mail messages - Avoid unit test txn rollback woes by making it possible to check for existence of a path with FileFolderService 30487: ALF-10268: Merged V3.4-BUG-FIX to PATCHES/V3.4.4 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 30540: ALF-10257: Fixed logic error introduced into Greenmail ImapRequestLineReader 30988: ALF-9361: Merged DEV/DAVEW/IMAP_NEW to PATCHES/V3.4.4 (by Arseny) 30419: Remote test for generic client request sequence. 30547: 1. A bug with FetchCommand particularly with FETCH (BODY.PEEK[1]) with an error 1315912197.789640 1.5 NO FETCH failed. java.lang.String cannot be cast to javax.mail.internet.MimeMultipart This happened while message content is being proceeded like MimeMultipart mp = (MimeMultipart) mimeMessage.getContent();, but javadoc of mimeMessage.getContent() says that this content can be a String in case of non-multipart message. Fixed FetchCommand accordingly to mimeMessage.getContent() javadoc. 2. A bug with RFC822MetadataExtracter When mimeMessage.getHeader("received"); is used with the message with following header Received: with ECARTIS (v1.0.0; list dovecot); Tue, 06 Aug 2002 13:01:17 +0300 (EEST) It doesn't extract a date, because it uses lastReceived.indexOf(';') which returns the position IN the ECARTIS (v1.0.0; list dovecot) after v1.0.0, So it should use lastReceived.lastIndexOf(';') to get the position after ECARTIS (v1.0.0; list dovecot). 31025: ALF-9361: IMAP Performance - Introduced folder status MRU cache - Keyed by user ID and change token so no need to cluster - Now means we should get reuse across IMAP sessions - Also fixed isMarked() implementation to only return true if there are recent or unseen mails 31038: ALF-9361: Prevent the starting of unnecessary transactions in AlfrescoImapFolder interface - getFolderStatus regulates its own transaction - Dropped all those *Internal methods from the abstract class - getUnqualifiedMailboxPattern moved to AlfrescoImapHostManager - Fixes to session folder cache validation / reuse 31039: ALF-9361: Repository tuning for IMAP performance - Backed out ALF-5575 60 second timeout on node caches - Should be covered by ALF-8607 fix - Also made TransactionalCache.NewCacheBucket save new values to the shared cache for 'mutable' caches. Previously it was only possibly to load into the node caches in a read only transaction! - Also added fix to make AbstractNodeDAOImpl bulk load empty node aspect sets - Result is a drastic speedup of full sync times as most items can be served from the cache` 31042: ALF-9361: Fix ImapServiceImplTest 31048: ALF-9361: Make ConcurrentNodeServiceTest work again, after relaxation of 'mutable' transactional caches - aspect and property caches validated by node transaction ID, as per parent assocs in ALF-8607 31050: ALF-9361: Caching correction Always use the cached mailbox reference if it is equivalent (because the session remembers the last selected mailbox) 31060: ALF-9361: Fix CacheTest, following back out of ALF-5575 behaviour 31061: ALF-9361: More caching fixes 31062: ALF-9361: Undo accidental changes to ConcurrentNodeServiceTest 31063: ALF-9361: Build fix: replaced assertSame with assertEquals git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@31079 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
775 lines
25 KiB
Java
775 lines
25 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.imap;
|
|
|
|
import java.io.IOException;
|
|
import java.io.Serializable;
|
|
import java.util.ArrayList;
|
|
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.MessagingException;
|
|
import javax.mail.Flags.Flag;
|
|
import javax.mail.internet.MimeMessage;
|
|
|
|
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.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.security.AccessStatus;
|
|
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;
|
|
|
|
private static Log logger = LogFactory.getLog(AlfrescoImapFolder.class);
|
|
|
|
/**
|
|
* 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;
|
|
|
|
/**
|
|
* 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, ServiceRegistry serviceRegistry)
|
|
{
|
|
this(null, userName, "", "", null, serviceRegistry, false, false);
|
|
}
|
|
|
|
|
|
/**
|
|
* Constructs {@link AlfrescoImapFolder} object.
|
|
*
|
|
* @param qualifiedMailboxName - name of the mailbox (e.g. "admin" for admin user).
|
|
* @param folderInfo - reference to the {@link FileInfo} object representing the folder.
|
|
* @param folderName - name of the folder.
|
|
* @param viewMode - defines view mode. Can be one of the following: {@link AlfrescoImapConst#MODE_ARCHIVE} or {@link AlfrescoImapConst#MODE_VIRTUAL}.
|
|
* @param rootNodeRef - reference to the root node of the store where folder is placed.
|
|
* @param mountPointName - name of the mount point.
|
|
*/
|
|
public AlfrescoImapFolder(
|
|
FileInfo folderInfo,
|
|
String userName,
|
|
String folderName,
|
|
String folderPath,
|
|
ImapViewMode viewMode,
|
|
boolean extractAttachmentsEnabled,
|
|
ServiceRegistry serviceRegistry)
|
|
{
|
|
this(folderInfo, userName, folderName, folderPath, viewMode, serviceRegistry, null, extractAttachmentsEnabled);
|
|
}
|
|
|
|
/**
|
|
* Constructs {@link AlfrescoImapFolder} object.
|
|
*
|
|
* @param qualifiedMailboxName - name of the mailbox (e.g. "admin" for admin user).
|
|
* @param folderInfo - reference to the {@link FileInfo} object representing the folder.
|
|
* @param folderName - name of the folder.
|
|
* @param viewMode - defines view mode. Can be one of the following: {@link AlfrescoImapConst#MODE_ARCHIVE} or {@link AlfrescoImapConst#MODE_VIRTUAL}.
|
|
* @param rootNodeRef - reference to the root node of the store where folder is placed.
|
|
* @param mountPointName - name of the mount point.
|
|
* @param imapService - reference to the {@link ImapHelper} object.
|
|
* @param selectable - defines whether the folder is selectable or not.
|
|
*/
|
|
public AlfrescoImapFolder(
|
|
FileInfo folderInfo,
|
|
String userName,
|
|
String folderName,
|
|
String folderPath,
|
|
ImapViewMode viewMode,
|
|
ServiceRegistry serviceRegistry,
|
|
Boolean selectable,
|
|
boolean extractAttachmentsEnabled)
|
|
{
|
|
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 = serviceRegistry.getImapService();
|
|
|
|
// 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;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* (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()
|
|
{
|
|
if (this.folderStatus == null)
|
|
{
|
|
CommandCallback<FolderStatus> command = new CommandCallback<FolderStatus>()
|
|
{
|
|
public FolderStatus command() throws Throwable
|
|
{
|
|
return imapService.getFolderStatus(userName, folderInfo.getNodeRef(), viewMode);
|
|
}
|
|
};
|
|
this.folderStatus = command.run();
|
|
}
|
|
return this.folderStatus;
|
|
}
|
|
|
|
/**
|
|
* 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 = createMimeMessageInFolder(this.folderInfo, message, flags);
|
|
// Invalidate current folder status
|
|
this.folderStatus = null;
|
|
return uid;
|
|
}
|
|
|
|
/**
|
|
* 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 void 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);
|
|
toImapMailFolder.appendMessageInternal(newMessage, imapService.getFlags(sourceMessageFileInfo), new Date());
|
|
}
|
|
else
|
|
{
|
|
serviceRegistry.getFileFolderService().copy(sourceMessageFileInfo.getNodeRef(), destFolderNodeRef, null);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* 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());
|
|
}
|
|
}
|
|
|
|
/**
|
|
* 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, false));
|
|
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;
|
|
}
|
|
};
|
|
command.run();
|
|
}
|
|
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;
|
|
}
|
|
|
|
/**
|
|
* 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);
|
|
}
|
|
|
|
/**
|
|
* 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(folderFileInfo.getNodeRef(), messageFile.getNodeRef(), message);
|
|
}
|
|
// Force persistence of the message to the repository
|
|
new IncomingImapMessage(messageFile, serviceRegistry, message);
|
|
return newMessageUid;
|
|
}
|
|
}
|