gecko-dev/lib/libmsg/msgdb.h

383 lines
17 KiB
C++

/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*-
*
* The contents of this file are subject to the Netscape Public
* License Version 1.1 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy of
* the License at http://www.mozilla.org/NPL/
*
* Software distributed under the License is distributed on an "AS
* IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
* implied. See the License for the specific language governing
* rights and limitations under the License.
*
* The Original Code is mozilla.org code.
*
* The Initial Developer of the Original Code is Netscape
* Communications Corporation. Portions created by Netscape are
* Copyright (C) 1998 Netscape Communications Corporation. All
* Rights Reserved.
*
* Contributor(s):
*/
#ifndef _MsgDB_H_
#define _MsgDB_H_
#include "msgcom.h"
#include "ptrarray.h"
#include "dberror.h"
#include "chngntfy.h"
#include "msgdbtyp.h"
class MessageDB;
class MsgDocument;
class DBFolderInfo;
class MailDB;
class NewsGroupDB;
class XPByteArray;
class msg_NewsArtSet;
class XPStringObj;
const int kMaxSubject = 160;
const int kMaxAuthor = 160;
const int kMaxRecipient = 80;
const int kMaxMsgIdLen = 80;
const int kMaxReferenceLen = 10 * kMaxMsgIdLen;
const MessageKey kIdNone = 0xffffffff;
const MessageKey kIdPending = 0xfffffffe;
const MessageKey kIdStartOfFake = 0xffffff80; // start of the fake message id
#include "idarray.h" // need MessageKey to be defined before including...
struct ListContext
{
MessageDB *m_pMessageDB;
int m_curMessageNum;
MSG_IteratorHandle m_iterator;
};
struct MessageHdrStruct
{
MessageKey m_threadId;
MessageKey m_messageKey;
char m_subject[kMaxSubject];
char m_author[kMaxAuthor];
char m_messageId[kMaxMsgIdLen];
char m_references[kMaxReferenceLen];
char m_recipients[kMaxRecipient];
time_t m_date;
uint32 m_messageSize; // lines for news articles, bytes for mail messages
uint32 m_flags;
uint16 m_numChildren; // for top-level threads
uint16 m_numNewChildren; // for top-level threads
char m_level; // indentation level
MSG_PRIORITY m_priority;
public:
void SetSubject(const char * subject);
void SetAuthor(const char * author);
void SetMessageID(const char * msgID);
void SetReferences(const char * referencesStr);
void SetDate(const char * date);
void SetLines(uint32 lines);
void SetSize(uint32 size);
const char *GetReference(const char *nextRef, char *reference);
static void StripMessageId(const char *msgID, char *outMsgId, int msgIdLen);
};
typedef int32 MsgFlags; // try to keep flags view is going to care about
// in the first byte, so we can copy the flags over directly
const int32 kIsRead = 0x1; // same as MSG_FLAG_READ
const int32 kReplied = 0x2; // same as MSG_FLAG_REPLIED
const int32 kMsgMarked = 0x4; // same as MSG_FLAG_MARKED - (kMarked collides with IMAP)
const int32 kHasChildren = 0x8; // no equivalent (shares space with FLAG_EXPUNGED)
const int32 kIsThread = 0x10; // !!shares space with MSG_FLAG_HAS_RE
const int32 kElided = 0x20; // same as MSG_FLAG_ELIDED
const int32 kExpired = 0x40; // same as MSG_FLAG_EXPIRED
const int32 kOffline = 0x80; // this message has been downloaded
const int32 kWatched = 0x100;
const int32 kSenderAuthed = 0x200; // same as MSG_FLAG_SENDER_AUTHED
const int32 kExpunged = 0x400; // NOT same value as MSG_FLAG_EXPUNGED
const int32 kHasRe = 0x800; // Not same values as MSG_FLAG_HAS_RE
const int32 kForwarded = 0x1000; // this message has been forward (mail only)
const int32 kIgnored = 0x2000; // this message is ignored
const int32 kMDNNeeded = 0x4000; // this message needs MDN report
const int32 kMDNSent = 0x8000; // MDN report has been Sent
const int32 kNew = 0x10000; // used for status, never stored in db.
const int32 kAdded = 0x20000; // Just added to db - used in
// notifications, never set in msgHdr.
const int32 kTemplate = 0x40000; // this message is a template
const int32 kDirty = 0x80000;
const int32 kPartial = 0x100000; // NOT same value as MSG_FLAG_PARTIAL
const int32 kIMAPdeleted = 0x200000; // same value as MSG_FLAG_IMAP_DELETED
const int32 kHasAttachment = 0x10000000; // message has attachments
const int32 kSameAsMSG_FLAG = kHasAttachment|kIsRead|kMsgMarked|kExpired|kElided|kSenderAuthed|kReplied|kOffline|kForwarded|kWatched|kIMAPdeleted;
const int32 kMozillaSameAsMSG_FLAG = kIsRead|kMsgMarked|kExpired|kElided|kSenderAuthed|kReplied|kOffline|kForwarded|kWatched|kIMAPdeleted;
const int32 kExtraFlags = 0xFF;
class MessageDBArray : public XPPtrArray
{
public:
MessageDBArray();
// overrides with proper types to avoid casting
MessageDB* GetAt(int nIndex) const {return((MessageDB*)XPPtrArray::GetAt(nIndex));}
void* operator[](int nIndex) const {return((MessageDB*)XPPtrArray::operator[](nIndex));}
};
#include "msghdr.h"
class DBThreadMessageHdr;
class DBMessageHdr;
typedef XP_Bool
HdrCompareFunc (DBMessageHdr *hdr);
// the DB doing threading, partly because we want it to be persistent,
// to avoid rethreading everytime a group is opened.
class MessageDB : public ChangeAnnouncer
{
friend class MessageDBView;
friend class DBThreadMessageHdr;
public:
MessageDB();
virtual ~MessageDB();
virtual MsgERR OpenDB(const char *dbFileName, XP_Bool create);
virtual MsgERR MessageDBOpen(const char * dbName, XP_Bool create);
virtual MsgERR Close();
virtual MsgERR ForceClosed();
virtual MsgERR CloseDB();
virtual MsgERR OnNewPath (const char * /*path*/) ;
static MsgERR RenameDB(const char *sourceName, const char *destName);
virtual MsgERR SetSummaryValid(XP_Bool valid = TRUE);
virtual MsgERR Commit(XP_Bool compress = FALSE);
virtual MsgERR GetMessageHdr(MessageKey messageNum, MessageHdrStruct *);
virtual MsgERR GetShortMessageHdr(MessageKey messageNum, MSG_MessageLine *msgHdr);
virtual XP_Bool KeyToAddExists(MessageKey messageKey);
virtual MsgERR AddHeaderToArray(DBMessageHdr *header);
virtual MsgERR AddHdr(DBMessageHdr *hdr);
virtual MsgERR AddHdrToDB(DBMessageHdr *newHdr, XP_Bool *newThread,
XP_Bool notify = FALSE);
virtual MsgERR AddThread(DBMessageHdr *msgHdr);
virtual DBThreadMessageHdr *GetDBThreadHdrForSubject(DBMessageHdr *msgHdr);
virtual DBThreadMessageHdr *GetDBMsgHdrForReference(const char * msgID);
virtual DBMessageHdr * GetDBMessageHdrForID(const char * msgID);
virtual MsgERR GetHeaderFromHandle(MSG_HeaderHandle headerHandle, DBMessageHdr **pResult);
DBThreadMessageHdr *GetThreadHeaderFromHandle(MSG_ThreadHandle handle);
virtual MessageKey GetMessageKeyForID(const char *msgID);
virtual DBThreadMessageHdr *GetDBThreadHdrForMsgID(MessageKey messageKey);
virtual MessageKey GetThreadIdForMsgId(MessageKey messageKey);
virtual MsgERR AddToThread(DBMessageHdr *reply, DBThreadMessageHdr *threadHdr, XP_Bool threadInThread);
// we've finished receiving new headers - clean up any temporary info.
virtual MsgERR FinishAddingHeaders();
// given an array of thread ids, return list header information for each thread
MsgERR ListHeaders(MessageKey *pMessageNums, int numToList, MessageHdrStruct *pOutput, int *pNumListed);
MsgERR ListHeadersShort(MessageKey *pMessageNums, int numToList, MSG_MessageLine *pOutput, int *pNumListed);
// list the headers of the top-level thread ids
virtual MsgERR ListThreadIds(MessageKey *startMsg, XP_Bool unreadOnly, MessageKey *pOutput, int32 *pFlags,
char *pLevels, int numToList, int *numListed, MessageDBView *view, int32 *pTotalHeaders);
virtual MsgERR ListThreadIds(ListContext *context, MessageKey *pOutput, int numToList, int *numListed);
// Info about a given thread
virtual MsgERR GetThreadCount(MessageKey messageKey, int32 *pThreadCount);
virtual MsgERR GetThreadCount(DBThreadMessageHdr *threadHdr, int32 *pThreadCount);
virtual MsgERR GetDBThreadListContext(MessageKey threadId, MessageKey startMsg, DBThreadMessageHdr **pThreadHdr, uint16 *pThreadIndex);
virtual MsgERR GetDBThreadListContext(DBMessageHdr *msgHdr, MessageKey startMsg, DBThreadMessageHdr **pThreadHdr, uint16 *pThreadIndex);
virtual DBThreadMessageHdr * GetDBThreadHdrForThreadID(MessageKey messageID);
virtual DBMessageHdr *GetDBHdrForKey(MessageKey messageKey);
virtual DBThreadMessageHdr *GetDBThreadHdrForMsgHdr(DBMessageHdr *msgHdr);
virtual MessageKey GetKeyOfFirstMsgInThread(MessageKey key);
// return a flat list of ids (no threading)
MsgERR ListAllIds(IDArray &outputIds);
MsgERR ListAllIds(IDArray *outputIds);
// iterate through message headers, in no particular order (currently, id order).
// Caller must unrefer return DBMessageHdr when done with it.
// eDBEndOfList will be returned when return DBMessageHdr * is NULL.
// Caller must call ListDone to free up the context.
// ListLast will produce a reverse iterator that starts at the end and moves towards the start
MsgERR ListFirst(ListContext **pContext, DBMessageHdr **pResult);
MsgERR ListLast(ListContext **pContext, DBMessageHdr **pResult);
MsgERR ListNext(ListContext *pContext, DBMessageHdr **pResult);
MsgERR ListDone(ListContext *pContext);
MessageKey ListHighwaterMark(); // returns the biggest key, kIdNone if none found.
// return the list header information for the documents in a thread.
virtual MsgERR ListIdsInThread(MessageKey threadId, MessageKey *startMsg, int numToList, MessageKey *pOutput, char *pFlags, char *pLevels, int *pNumListed);
virtual MsgERR ListIdsInThread(DBMessageHdr *msgHdr, MessageKey *startMsg, int numToList, MessageKey *pOutput, char *pFlags, char *pLevels, int *pNumListed);
virtual MsgERR ListIdsInThread(DBThreadMessageHdr *threadHdr, uint16 threadIndex, MessageKey *startMsg, int numToList, MessageKey *pOutput, char *pFlags, char *pLevels, int *pNumListed);
virtual MsgERR ListUnreadIdsInThread(MessageKey threadId, MessageKey *startMsg, XPByteArray &levelStack, int numToList, MessageKey *pOutput, char *pFlags, char *pLevels, int *pNumListed);
virtual MsgERR ListNextUnread(ListContext **pContext, DBMessageHdr **pResult);
static XP_Bool MatchFlaggedNotOffline(DBMessageHdr *hdr);
virtual void ListMatchingKeys(HdrCompareFunc *compareFunc, IDArray &matchingKeys);
MsgERR Purge();
// MDN support
virtual MsgERR MarkMDNNeeded(MessageKey messageKey, XP_Bool bNeeded,
ChangeListener *instigator = NULL);
// MarkMDNneeded only used when mail server is a POP3 server
// or when the IMAP server does not support user defined
// PERMANENTFLAGS
virtual MsgERR IsMDNNeeded(MessageKey messageKey, XP_Bool *isNeeded);
virtual MsgERR MarkMDNSent(MessageKey messageKey, XP_Bool bNeeded,
ChangeListener *instigator = NULL);
virtual MsgERR IsMDNSent(MessageKey messageKey, XP_Bool *isSent);
// methods to get and set docsets for ids.
virtual MsgERR MarkRead(MessageKey messageKey, XP_Bool bRead,
ChangeListener *instigator = NULL);
virtual MsgERR MarkReplied(MessageKey messageKey, XP_Bool bReplied,
ChangeListener *instigator = NULL);
virtual MsgERR MarkForwarded(MessageKey messageKey, XP_Bool bForwarded,
ChangeListener *instigator = NULL);
virtual MsgERR MarkHasAttachments(MessageKey messageKey, XP_Bool bHasAttachments,
ChangeListener *instigator = NULL);
virtual MsgERR MarkThreadIgnored(DBThreadMessageHdr *thread, MessageKey threadKey, XP_Bool bIgnored,
ChangeListener *instigator = NULL);
virtual MsgERR MarkThreadWatched(DBThreadMessageHdr *thread, MessageKey threadKey, XP_Bool bWatched,
ChangeListener *instigator = NULL);
virtual MsgERR IsRead(MessageKey messageKey, XP_Bool *pRead);
virtual MsgERR IsIgnored(MessageKey messageKey, XP_Bool *pIgnored);
virtual MsgERR IsMarked(MessageKey messageKey, XP_Bool *pMarked);
virtual MsgERR HasAttachments(MessageKey messageKey, XP_Bool *pHasThem);
virtual MsgERR DeleteMessages(IDArray &messageKeys, ChangeListener *instigator);
virtual MsgERR DeleteMessage(MessageKey messageKey,
ChangeListener *instigator = NULL,
XP_Bool commit = TRUE);
virtual MsgERR DeleteHeader(DBMessageHdr *msgHdr, ChangeListener *instigator, XP_Bool commit, XP_Bool onlyRemoveFromThread = FALSE);
virtual MsgERR UndoDelete(DBMessageHdr *msgHdr);
virtual MsgERR MarkLater(MessageKey messageKey, time_t until);
virtual MsgERR MarkMarked(MessageKey messageKey, XP_Bool mark,
ChangeListener *instigator = NULL);
virtual MsgERR MarkOffline(MessageKey messageKey, XP_Bool offline,
ChangeListener *instigator);
virtual XP_Bool AllMessageKeysImapDeleted(const IDArray &keys);
virtual MsgERR MarkImapDeleted(MessageKey messageKey, XP_Bool deleted,
ChangeListener *instigator);
virtual void RemoveHeaderFromDB(DBMessageHdr *msgHdr);
virtual MsgERR GetUnreadKeyInThread(MessageKey threadId,
MessageKey *resultKey,
MessageKey *resultThreadId);
virtual MsgERR MarkReadByDate (time_t startDate, time_t endDate, MWContext *context, IDArray *markedIds);
virtual MsgERR MarkAllRead(MWContext *context, IDArray *thoseMarked = NULL);
virtual XP_Bool SetPriority(MessageKey messageKey, MSG_PRIORITY priority);
virtual XP_Bool SetPriority(DBMessageHdr *msgHdr, MSG_PRIORITY priority);
void SetSortInfo(SortType, SortOrder);
MsgERR GetSortInfo(SortType *, SortOrder *);
virtual void HandleLatered(); // mark latered documents unread
// methods to handle new message support
MessageKey GetFirstNew();
void AddToNewList(MessageKey key);
XP_Bool HasNew();
void ClearNewList(XP_Bool notify = FALSE);
MessageKey GetUnusedFakeId();
DBFolderInfo *m_dbFolderInfo;
MSG_DBHandle GetDB() {return m_dbHandle;}
// typesafe downcasts for db's
virtual MailDB *GetMailDB(); // returns NULL if not a mail db
virtual NewsGroupDB *GetNewsDB(); // returns NULL if not a news db
virtual MSG_FolderInfo *GetFolderInfo();
void SetFolderInfo(MSG_FolderInfo *info) {m_folderInfo = info;}
virtual int GetCurVersion() {return 0;}
DBFolderInfo *GetDBFolderInfo() {return m_dbFolderInfo;}
int AddUseCount() {return ++m_useCount;}
static void ConvertDBFlagsToPublicFlags(uint32 *flags);
static void ConvertPublicFlagsToDBFlags(uint32 *flags);
static MessageDB* FindInCache(const char * pDBName);
static void CopyFullHdrToShortHdr(MSG_MessageLine *msgHdr,
MessageHdrStruct *fullHdr);
static void CleanupCache();
#ifdef DEBUG
static int GetNumInCache(void) {return(GetDBCache()->GetSize());}
static void DumpCache();
#endif
ViewType GetViewType();
void SetViewType(ViewType viewType);
uint32 GetStatusFlags(DBMessageHdr *msgHdr);
#ifdef DEBUG_bienvenu
void Verify();
#endif
virtual MsgERR MarkHdrRead(DBMessageHdr *msgHdr, XP_Bool bRead,
ChangeListener *instigator);
virtual MsgERR GetCachedPassword(XPStringObj &cachedPassword);
virtual MsgERR SetCachedPassword(const char *password);
virtual XP_Bool HasCachedPassword();
protected:
virtual MsgERR SetKeyFlag(MessageKey messageKey, XP_Bool set, int32 flag,
ChangeListener *instigator /* = NULL */);
virtual void SetHdrFlag(DBMessageHdr *, XP_Bool bSet, MsgFlags flag);
virtual void MarkHdrReadInDB(DBMessageHdr *msgHdr, XP_Bool bRead,
ChangeListener *instigator);
virtual DBFolderInfo *CreateFolderInfo(MSG_DBFolderInfoHandle handle);
virtual MsgERR IsHeaderRead(DBMessageHdr *hdr, XP_Bool *pRead);
XP_Bool MatchDbName(const char * dbName); // returns TRUE if they match
DBFolderInfo *AddNewFolderInfoToDB();
// helper function for creating iterators;
MsgERR CreateListIterator(XP_Bool forward, ListContext **pContext, DBMessageHdr **pResult);
MsgERR GetMessageIndexInThread(DBThreadMessageHdr *threadHdr, MessageKey startMsg, uint16 *pThreadIndex);
protected:
MSG_DBHandle m_dbHandle;
MSG_DBFolderInfoHandle m_dbFolderInfoHandle;
MSG_IteratorHandle m_threadIterator;
char *m_dbName;
int m_useCount;
MSG_FolderInfo *m_folderInfo;
XPPtrArray m_newHeaders; // array of DBMessageHdr *
int m_headerIndex; // index of last header added to m_newHeaders
int32 m_addCount; // how many added since last commit
int32 m_commitChunk; // how many to add before committing
IDArray m_lockedKeys;
msg_NewsArtSet *m_newSet; // new messages since last open.
// static message db cache functions - perhaps overlaps with URL caching
// since we can open message dbs without opening a UI on the message db,
// or have two ui widgets open the same db.
static MessageDBArray *m_dbCache; // array of messageDB's
static XP_Bool m_cacheEnabled;
static void AddToCache(MessageDB* pMessageDB)
{GetDBCache()->Add(pMessageDB);}
static void RemoveFromCache(MessageDB* pMessageDB);
static int FindInCache(MessageDB* pMessageDB);
static XP_Bool EnableCache(XP_Bool enable);
static XP_Bool IsCacheEnabled(void) {return(m_cacheEnabled);}
static MessageDBArray* GetDBCache();
};
#endif