/* -*- 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): */ #include "msg.h" #include "xp.h" #include "xp_time.h" #include "msgdb.h" #include "msgdbvw.h" #include "dberror.h" #include "grpinfo.h" #include "thrhead.h" #include "xpgetstr.h" #include "newsset.h" #include "msgdbapi.h" extern "C" { extern int MK_MSG_MARKREAD_COUNT; extern int MK_MSG_DONE_MARKREAD_COUNT; } XP_Bool MessageDB::m_cacheEnabled = TRUE; MessageDBArray /*NEAR*/ *MessageDB::m_dbCache = NULL; MessageDBArray::MessageDBArray() { } MessageDB::MessageDB() { m_useCount = 0; m_dbHandle = NULL; m_headerIndex = 0; m_addCount = 0; m_commitChunk = 200; m_dbName = NULL; m_dbFolderInfo = NULL; m_newSet = NULL; m_folderInfo = NULL; } //---------------------------------------------------------------------- // GetDBCache //---------------------------------------------------------------------- MessageDBArray * MessageDB::GetDBCache() { if (!m_dbCache) { m_dbCache = new MessageDBArray(); } return m_dbCache; } void MessageDB::CleanupCache() { if (m_dbCache) // clean up memory leak { for (int i = 0; i < GetDBCache()->GetSize(); i++) { MessageDB* pMessageDB = GetDBCache()->GetAt(i); if (pMessageDB) { #ifdef DEBUG_bienvenu XP_Trace("closing %s\n", pMessageDB->m_dbName); #endif pMessageDB->ForceClosed(); i--; // back up array index, since closing removes db from cache. } } XP_ASSERT(GetNumInCache() == 0); // better not be any open db's. delete m_dbCache; } m_dbCache = NULL; // Need to reset to NULL since it's a // static global ptr and maybe referenced // again in other places. } MessageDB::~MessageDB() { NotifyAnnouncerGoingAway(NULL); Purge(); if (m_dbName) XP_FREE(m_dbName); if (m_newSet) delete m_newSet; } MsgERR MessageDB::MessageDBOpen(const char * dbName, XP_Bool create) { MsgERR err = OpenDB(dbName, create); if (err == eSUCCESS) m_useCount++; return err; } MsgERR MessageDB::Close() { --m_useCount; XP_ASSERT(m_useCount >= 0); if (m_useCount == 0) { Purge(); #ifdef DEBUG_bienvenu1 Verify(); #endif CloseDB(); RemoveFromCache(this); #ifdef DEBUG_bienvenu if (GetNumInCache() != 0) { XP_Trace("closing %s\n", m_dbName); DumpCache(); } #endif // if this terrifies you, we can make it a static method delete this; return(eSUCCESS); } else { return(eSUCCESS); } } // virtual inlines moved to .cpp file to help compiler charity cases. MsgERR MessageDB::OnNewPath (const char * /*path*/) { return eSUCCESS; } // force the database to close - this'll flush out anybody holding onto // a database without having a listener! MsgERR MessageDB::ForceClosed() { MsgERR err = eSUCCESS; while (m_useCount > 0 && err == eSUCCESS) { int32 saveUseCount = m_useCount; err = Close(); if (saveUseCount == 1) break; } return err; } // this routine should not leave the database open if it returns an error. MsgERR MessageDB::OpenDB(const char *dbFileName, XP_Bool create) { MsgERR err = MSG_OpenDB(dbFileName, create, &m_dbHandle, &m_dbFolderInfoHandle); XPStringObj newSet; if (err == eSUCCESS) { MSG_DBFolderInfoExchange exchangeInfo; if (!m_dbFolderInfoHandle) AddNewFolderInfoToDB(); if (m_dbName) XP_FREE(m_dbName); m_dbName = XP_STRDUP( dbFileName ); m_dbFolderInfo = CreateFolderInfo(m_dbFolderInfoHandle); m_dbFolderInfo->SetHandle(m_dbFolderInfoHandle); MSG_DBFolderInfo_GetFolderInfo(m_dbFolderInfoHandle, &exchangeInfo); m_dbFolderInfo->SetExchangeInfo(exchangeInfo); // compare db version filed out to current db version if (m_dbFolderInfoHandle != NULL && GetCurVersion() != m_dbFolderInfo->GetDiskVersion()) { CloseDB(); err = eOldSummaryFile; } else { m_dbFolderInfo->GetNewArtsSet(newSet); m_newSet = msg_NewsArtSet::Create(newSet); } } return err; } MsgERR MessageDB::CloseDB() { if (m_dbHandle != NULL) { Commit(); if (m_dbFolderInfo != NULL) { delete m_dbFolderInfo; m_dbFolderInfo = NULL; } MSG_CloseDB(m_dbHandle); } return eSUCCESS; } DBFolderInfo *MessageDB::AddNewFolderInfoToDB() { m_dbFolderInfo = CreateFolderInfo(0); // m_dbFolderInfo->fID = 1; // one and only newsgroup info m_dbFolderInfo->SetHighWater(0); m_dbFolderInfo->m_version = GetCurVersion(); // m_dbFolderInfo->setDirty(); MSG_AddDBFolderInfo(m_dbHandle, m_dbFolderInfo->GetHandle()); m_dbFolderInfoHandle = m_dbFolderInfo->GetHandle(); Commit(); return m_dbFolderInfo; } // Renames the destdb as sourceDB. DestDB could be open, in which case we need to rename it under // the covers. SourceDB better not be open, since it's going away. /*static*/ MsgERR MessageDB::RenameDB(const char *sourceName, const char *destName) { MsgERR err = eSUCCESS; #ifdef DEBUG { char* filename = WH_FileName(sourceName, xpMailFolderSummary); XP_ASSERT(filename); XP_ASSERT(MessageDB::FindInCache(filename) == NULL); FREEIF(filename); } #endif MessageDB *destDB; { char* filename = WH_FileName(destName, xpMailFolderSummary); if (!filename) return eOUT_OF_MEMORY; destDB = MessageDB::FindInCache(filename); XP_FREE(filename); } if (destDB) err = destDB->CloseDB(); if (XP_FileRename(sourceName, xpMailFolderSummary, destName, xpMailFolderSummary) == 0) { if (destDB) { char* filename = WH_FileName (destName, xpMailFolderSummary); if (!filename) return eOUT_OF_MEMORY; err = destDB->OpenDB (filename, FALSE /*create?*/); XP_FREE(filename); } } else err = eFAILURE; // ### dmb - need to come up with better error. return err; } MsgERR MessageDB::SetSummaryValid(XP_Bool valid /* = TRUE */) { if (!valid) { if (m_dbFolderInfo) { m_dbFolderInfo->m_version = 0; // DMB TODO m_dbFolderInfo->setDirty(); Commit(); } else return eFAILURE; // can't do this w/o a folder info... } // for default db (and news), there's no nothing to set to make it it valid return eSUCCESS; } // returns NULL if not a mail db MailDB *MessageDB::GetMailDB() { return NULL; } // returns NULL if not a news db NewsGroupDB *MessageDB::GetNewsDB() { return NULL; } DBFolderInfo *MessageDB::CreateFolderInfo(MSG_DBFolderInfoHandle handle) { return new DBFolderInfo((handle) ? handle : MSG_CreateMailDBFolderInfo()); } MSG_FolderInfo *MessageDB::GetFolderInfo() { return m_folderInfo; } MsgERR MessageDB::GetHeaderFromHandle(MSG_HeaderHandle headerHandle, DBMessageHdr **pResult) { DBMessageHdr *header = new DBMessageHdr(headerHandle); *pResult = header; return eSUCCESS; } DBThreadMessageHdr *MessageDB::GetThreadHeaderFromHandle(MSG_ThreadHandle handle) { return (handle) ? new DBThreadMessageHdr(handle) : 0; } MsgERR MessageDB::Commit(XP_Bool compress /* = FALSE */) { MSG_DBFolderInfoExchange exchangeInfo; XP_ASSERT(m_dbHandle != NULL); char *outputLine = (m_newSet) ? m_newSet->Output() : 0; if (outputLine) { GetDBFolderInfo()->SetNewArtsSet(outputLine, m_dbHandle); delete [] outputLine; } else GetDBFolderInfo()->SetNewArtsSet("", m_dbHandle); m_dbFolderInfo->GetExchangeInfo(exchangeInfo); MSG_DBFolderInfo_SetFolderInfo(m_dbFolderInfoHandle, &exchangeInfo, m_dbHandle); MsgERR err = MSG_CommitDB(m_dbHandle, compress); if (err != eSUCCESS) return eWRITE_ERROR; return eSUCCESS; } // Fill in messageHdr from database. Don't change msgHdr if we don't find messageNum. MsgERR MessageDB::GetMessageHdr(MessageKey messageKey, MessageHdrStruct *msgHdr) { DBMessageHdr *headerObject = GetDBHdrForKey(messageKey); if (headerObject == NULL) return 0; else { XP_Bool isRead = FALSE; headerObject->CopyToMessageHdr(msgHdr, GetDB()); // unless/until marking read/unread goes through view layer, we need to get unreadness from newsrc IsRead(messageKey, &isRead); if (isRead) msgHdr->m_flags |= kIsRead; delete headerObject; return eSUCCESS; } } MsgERR MessageDB::GetShortMessageHdr(MessageKey messageNum, MSG_MessageLine *msgHdr) { MessageHdrStruct messageHdr; XP_MEMSET(&messageHdr, 0, sizeof(messageHdr)); DBMessageHdr *headerObject = GetDBHdrForKey(messageNum); if (headerObject == NULL) { return eID_NOT_FOUND; } else { XP_Bool isRead = FALSE; headerObject->CopyToShortMessageHdr(msgHdr, GetDB()); // unless/until marking read/unread goes through view layer, we need to get unreadness from newsrc IsRead(messageNum, &isRead); if (isRead) msgHdr->flags |= kIsRead; if (m_newSet && m_newSet->IsMember(headerObject->GetMessageKey())) msgHdr->flags |= kNew; delete headerObject; return eSUCCESS; } } // This function merely takes a DBMessageHdr // and stores that in m_newHeaders. From there, it will be threaded // and added to the database in a separate pass. MsgERR MessageDB::AddHeaderToArray(DBMessageHdr *dbMsgHdr) { m_newHeaders.SetAtGrow(m_headerIndex++, dbMsgHdr); return eSUCCESS; } // given an array of message ids, return list header information for each header MsgERR MessageDB::ListHeaders(MessageKey *pMessageNums, int numToList, MessageHdrStruct *pOutput, int *pNumListed) { MsgERR err = eSUCCESS; *pNumListed = 0; for (int i = 0; i < numToList; i++) { err = GetMessageHdr(pMessageNums[i], &pOutput[i]); if (err != eSUCCESS) break; (*pNumListed)++; } return err; } MsgERR MessageDB::ListHeadersShort(MessageKey *pMessageNums, int numToList, MSG_MessageLine *pOutput, int *pNumListed) { MsgERR err = eSUCCESS; *pNumListed = 0; for (int i = 0; i < numToList; i++) { err = GetShortMessageHdr(pMessageNums[i], &pOutput[i]); if (err != eSUCCESS) break; (*pNumListed)++; } return err; } MsgERR MessageDB::ListNextUnread(ListContext **pContext, DBMessageHdr **pResult) { DBMessageHdr *pHeader; MsgERR dbErr = eSUCCESS; XP_Bool lastWasRead = TRUE; *pResult = NULL; while (TRUE) { if (*pContext == NULL) dbErr = ListFirst (pContext, &pHeader); else dbErr = ListNext(*pContext, &pHeader); if (dbErr != eSUCCESS) { ListDone(*pContext); break; } // this currently doesn't happen since ListNext doesn't return errors // other than eDBEndOfList. else if (dbErr != eSUCCESS) break; if (IsHeaderRead(pHeader, &lastWasRead) == eSUCCESS && !lastWasRead) break; else delete pHeader; } if (!lastWasRead) *pResult = pHeader; return dbErr; } MsgERR MessageDB::ListAllIds(IDArray *outputIds) { MessageKey *resultKeys; int32 numKeys; MsgERR err = MSG_DBHandle_ListAllKeys(m_dbHandle, &resultKeys, &numKeys); outputIds->SetArray(resultKeys, numKeys, numKeys); return err; } MsgERR MessageDB::ListAllIds(IDArray &outputIds) { return ListAllIds(&outputIds); } MsgERR MessageDB::ListFirst(ListContext **pContext, DBMessageHdr **pResult) { MsgERR err; err = CreateListIterator(TRUE, pContext, pResult); if (err == eCorruptDB) { err = eEXCEPTION; SetSummaryValid(FALSE); } return err; } MsgERR MessageDB::ListNext(ListContext *pContext, DBMessageHdr **pResult) { DBMessageHdr *msgHdr = NULL; MsgERR err; MSG_HeaderHandle headerHandle; err = MSG_IteratorHandle_GetNextHeader(pContext->m_iterator, m_dbHandle, &headerHandle); if (err == eSUCCESS) err = GetHeaderFromHandle(headerHandle, pResult); else if (err == eCorruptDB) SetSummaryValid(FALSE); return err; } MsgERR MessageDB::ListLast(ListContext **pContext, DBMessageHdr **pResult) { return CreateListIterator(FALSE, pContext, pResult); } // returns the biggest key, kIdNone if none found. MessageKey MessageDB::ListHighwaterMark() { return MSG_DBHandle_GetHighwaterMark(m_dbHandle); } // Create an iterator that either starts at the beginning and goes towards the end // (if forward == TRUE), or starts at the end and goes towards the beginning. MsgERR MessageDB::CreateListIterator(XP_Bool forward, ListContext **pContext, DBMessageHdr **pResult) { MSG_HeaderHandle headerHandle; ListContext *listContext = NULL; listContext = new ListContext; MsgERR err = MSG_DBHandle_CreateHdrListIterator(m_dbHandle, forward, &listContext->m_iterator, &headerHandle); if (err == eSUCCESS) err = GetHeaderFromHandle(headerHandle, pResult); *pContext = listContext; return err; } MsgERR MessageDB::ListDone(ListContext *pContext) { if (pContext != NULL) { if (pContext->m_iterator) MSG_IteratorHandle_DestroyIterator(pContext->m_iterator); delete pContext; } return eSUCCESS; } /* static */ XP_Bool MessageDB::MatchFlaggedNotOffline(DBMessageHdr *hdr) { return (hdr->GetFlags() & kMsgMarked) && !(hdr->GetFlags() & kOffline); } void MessageDB::ListMatchingKeys(HdrCompareFunc *compareFunc, IDArray &matchingKeys) { MsgERR dbErr; DBMessageHdr *pHeader; ListContext *listContext = NULL; while (TRUE) { if (listContext == NULL) dbErr = ListFirst (&listContext, &pHeader); else dbErr = ListNext(listContext, &pHeader); if (dbErr == eDBEndOfList) { dbErr = eSUCCESS; ListDone(listContext); break; } // this currently doesn't happen since ListNext doesn't return errors // other than eDBEndOfList. else if (dbErr != eSUCCESS) break; if ((*compareFunc)(pHeader)) matchingKeys.Add(pHeader->GetMessageKey()); delete pHeader; } } // list the ids of the top-level thread ids starting at id == startMsg. This actually returns // the ids of the first message in each thread. MsgERR MessageDB::ListThreadIds(MessageKey *startMsg, XP_Bool unreadOnly, MessageKey *pOutput, int32 *pFlags, char *pLevels, int numToList, int *pNumListed, MessageDBView *view, int32 *pTotalHeaders) { MsgERR err = eSUCCESS; DBMessageHdr *msgHdr; MSG_ThreadHandle threadHandle = NULL; DBThreadMessageHdr * threadHdr = NULL; // N.B..don't ret before assigning numListed to *pNumListed int numListed = 0; if (*startMsg > 0) { XP_ASSERT(m_threadIterator != NULL); // for now, we'll just have to rely on the caller leaving // the iterator in the right place. err = MSG_IteratorHandle_GetNextThread(m_threadIterator, m_dbHandle, &threadHandle); } else { MSG_DBHandle_CreateThreadListIterator(m_dbHandle, TRUE, &m_threadIterator, &threadHandle); } if (!threadHandle) { if (*startMsg > 0) err = eID_NOT_FOUND; } else { threadHdr = new DBThreadMessageHdr(threadHandle); int32 threadCount; int32 threadsRemoved = 0; for (int i = 0; i < numToList && threadHdr != NULL; i++) { MSG_ThreadHandle nextThreadHandle = NULL; DBThreadMessageHdr *nextThreadHdr = NULL; err = MSG_IteratorHandle_GetNextThread(m_threadIterator, m_dbHandle, &nextThreadHandle); if (err == eCorruptDB) break; else if (err == eDBEndOfList) err = eSUCCESS; if (nextThreadHandle) nextThreadHdr = new DBThreadMessageHdr(nextThreadHandle); else nextThreadHdr = NULL; if (threadHdr->GetNumChildren() != 0) // not empty thread { if (pTotalHeaders) *pTotalHeaders += threadHdr->GetNumChildren(); if (unreadOnly) msgHdr = threadHdr->GetFirstUnreadChild(this); else msgHdr = threadHdr->GetChildHdrAt(0); uint32 threadFlags = threadHdr->GetFlags(); if (msgHdr != NULL && (!view || view->WantsThisThread(threadHdr))) { pOutput[numListed] = msgHdr->GetMessageKey(); pLevels[numListed] = 0 /* msgHdr->GetLevel() */; // DMB TODO - This will do for now...Until we decide how to // handle thread flags vs. message flags, if we do decide // to make them different. msgHdr->OrFlags(threadFlags & (kWatched | kIgnored)); XP_Bool isRead = FALSE; // make sure DB agrees with newsrc, if we're news. IsRead(msgHdr->GetMessageKey(), &isRead); MarkHdrRead(msgHdr, isRead, NULL); // try adding in kIsThread flag for unreadonly view. if (GetThreadCount(threadHdr, &threadCount) == eSUCCESS && threadCount > 1) pFlags[numListed] |= kHasChildren; pFlags[numListed] = msgHdr->m_flags | kIsThread | threadFlags; numListed++; } delete msgHdr; } else if (threadsRemoved < 10 && !(threadHdr->GetFlags() & (kWatched | kIgnored))) { MSG_DBHandle_RemoveThread(m_dbHandle, threadHandle); threadsRemoved++; // don't want to remove all empty threads first time // around as it will choke preformance for upgrade. #ifdef DEBUG_bienvenu XP_Trace("removing empty non-ignored non-watched thread\n"); #endif } delete threadHdr; threadHdr = nextThreadHdr; } } if (threadHdr != NULL) { *startMsg = threadHdr->GetThreadID(); delete threadHdr; } else { *startMsg = kIdNone; MSG_IteratorHandle_DestroyIterator(m_threadIterator); m_threadIterator = NULL; } *pNumListed = numListed; return err; } // helper function to get the thread list context from a thread id and start msg. // If successful, pThreadHdr will be non null on return MsgERR MessageDB::GetDBThreadListContext(MessageKey threadId, MessageKey startMsg, DBThreadMessageHdr **pThreadHdr, uint16 *pThreadIndex) { *pThreadIndex = 0; *pThreadHdr = GetDBThreadHdrForThreadID(threadId); if (*pThreadHdr == NULL) return eID_NOT_FOUND; return GetMessageIndexInThread(*pThreadHdr, startMsg, pThreadIndex); } MsgERR MessageDB::GetMessageIndexInThread(DBThreadMessageHdr *threadHdr, MessageKey startMsg, uint16 *pThreadIndex) { XP_Bool foundStartMsg = FALSE; if (startMsg != kIdNone && startMsg != 0) { while (*pThreadIndex < threadHdr->GetNumChildren()) { if (threadHdr->GetChildAt(*pThreadIndex) == startMsg) { foundStartMsg = TRUE; break; } (*pThreadIndex)++; } } else { *pThreadIndex = 0; foundStartMsg = TRUE; } return (foundStartMsg) ? eSUCCESS : eID_NOT_FOUND; } // overloaded helper function to get the thread list context from a msgHdr and start msg. // If successful, pThreadHdr will be non null on return MsgERR MessageDB::GetDBThreadListContext(DBMessageHdr *msgHdr, MessageKey startMsg, DBThreadMessageHdr **pThreadHdr, uint16 *pThreadIndex) { *pThreadIndex = 0; *pThreadHdr = GetDBThreadHdrForMsgHdr(msgHdr); if (*pThreadHdr == NULL) return eID_NOT_FOUND; return GetMessageIndexInThread(*pThreadHdr, startMsg, pThreadIndex); } // Overloaded ListThreadIds which takes a ListContext MsgERR MessageDB::ListThreadIds(ListContext * /*context*/, MessageKey * /*pOutput*/, int /*numToList*/, int * /*numListed*/) { return eNYI; } MsgERR MessageDB::ListUnreadIdsInThread(MessageKey threadId, MessageKey *startMsg, XPByteArray &levelStack, int numToList, MessageKey *pOutput, char *pFlags, char *pLevels, int *pNumListed) { MsgERR err; uint16 threadIndex = 0; DBThreadMessageHdr *threadHdr; *pNumListed = 0; err = GetDBThreadListContext(threadId, *startMsg, &threadHdr, &threadIndex); if (err != eSUCCESS) return err; // these children ids are in thread order. int i; int startNumListed = *pNumListed; for (i = 0; i + threadIndex < threadHdr->GetNumChildren() && (*pNumListed - startNumListed) < numToList; i++) { DBMessageHdr *msgHdr = threadHdr->GetChildHdrAt(i + threadIndex); if (msgHdr != NULL) { // if the current header's level is <= to the top of the level stack, // pop off the top of the stack. // ### dmb unreadonly - The level stack needs to work across calls // to this routine, in the case that we have more than 200 unread // messages in a thread. while (levelStack.GetSize() > 1 && msgHdr->GetLevel() <= levelStack.GetAt(levelStack.GetSize() - 1)) { levelStack.RemoveAt(levelStack.GetSize() - 1); } if (! (msgHdr->GetFlags() & kExpunged)) { XP_Bool isRead = FALSE; IsRead(msgHdr->GetMessageKey(), &isRead); if (!isRead) { uint8 levelToAdd; // just make sure flag is right in db. MarkHdrRead(msgHdr, FALSE, NULL); *pOutput++ = msgHdr->GetMessageKey(); // was fId DMB TODO *pFlags = 0; // pLevels[i] = msgHdr->GetLevel(); if (levelStack.GetSize() == 0) levelToAdd = 0; else levelToAdd = levelStack.GetAt(levelStack.GetSize() - 1) + 1; *pLevels++ = levelToAdd; #ifdef DEBUG_bienvenu // XP_Trace("added at level %d\n", levelToAdd); #endif levelStack.Add(levelToAdd); MessageDBView::CopyDBFlagsToExtraFlags(msgHdr->m_flags, pFlags); pFlags++; (*pNumListed)++; } } delete msgHdr; } } if ((i + threadIndex) < threadHdr->GetNumChildren()) *startMsg = threadHdr->GetChildAt(i + threadIndex); else *startMsg = kIdNone; delete threadHdr; return eSUCCESS; } MsgERR MessageDB::ListIdsInThread(DBMessageHdr *msgHdr, MessageKey *startMsg, int numToList, MessageKey *pOutput, char *pFlags, char *pLevels, int *pNumListed) { uint16 threadIndex = 0; DBThreadMessageHdr *threadHdr; MsgERR err; err = GetDBThreadListContext(msgHdr, *startMsg, &threadHdr, &threadIndex); if (err != eSUCCESS) return err; return ListIdsInThread(threadHdr, threadIndex, startMsg, numToList, pOutput, pFlags, pLevels, pNumListed); } MsgERR MessageDB::ListIdsInThread(MessageKey threadId, MessageKey *startMsg, int numToList, MessageKey *pOutput, char *pFlags, char *pLevels, int *pNumListed) { uint16 threadIndex = 0; DBThreadMessageHdr *threadHdr; MsgERR err; *pNumListed = 0; err = GetDBThreadListContext(threadId, *startMsg, &threadHdr, &threadIndex); if (err != eSUCCESS) return err; return ListIdsInThread(threadHdr, threadIndex, startMsg, numToList, pOutput, pFlags, pLevels, pNumListed); } MsgERR MessageDB::ListIdsInThread(DBThreadMessageHdr *threadHdr, uint16 threadIndex, MessageKey *startMsg, int numToList, MessageKey *pOutput, char *pFlags, char *pLevels, int *pNumListed) { // these children ids should be in thread order. int i; *pNumListed = 0; for (i = 0; i + threadIndex < threadHdr->GetNumChildren() && i < numToList; i++) { DBMessageHdr *msgHdr = threadHdr->GetChildHdrAt(i + threadIndex); if (msgHdr != NULL) { if (! (msgHdr->GetFlags() & kExpunged)) { XP_Bool isRead = FALSE; IsRead(msgHdr->GetMessageKey(), &isRead); // just make sure flag is right in db. MarkHdrRead(msgHdr, isRead, NULL); // if (isRead) // msgHdr->m_flags |= kIsRead; // else // msgHdr->m_flags &= ~kIsRead; *pOutput++ = msgHdr->GetMessageKey(); pFlags[i] = 0; pLevels[i] = msgHdr->GetLevel(); // turn off thread or elided bit if they got turned on (maybe from new only view?) if (i > 0) msgHdr->AndFlags(~(kIsThread|kElided)); MessageDBView::CopyDBFlagsToExtraFlags(msgHdr->m_flags, &pFlags[i]); (*pNumListed)++; } else { XP_ASSERT(FALSE); // shouldn't happen - expunging should remove } delete msgHdr; } } if ((i + threadIndex) < threadHdr->GetNumChildren()) *startMsg = threadHdr->GetChildAt(i + threadIndex); else *startMsg = kIdNone; delete threadHdr; return eSUCCESS; } MsgERR MessageDB::GetThreadCount(MessageKey messageKey, int32 *pThreadCount) { MsgERR ret = eID_NOT_FOUND; DBThreadMessageHdr *threadHdr = GetDBThreadHdrForMsgID(messageKey); if (threadHdr != NULL) { ret = GetThreadCount(threadHdr, pThreadCount); delete threadHdr; } return ret; } MsgERR MessageDB::GetThreadCount(DBThreadMessageHdr *threadHdr, int32 *pThreadCount) { *pThreadCount = threadHdr->GetNumChildren(); return eSUCCESS; } MessageKey MessageDB::GetKeyOfFirstMsgInThread(MessageKey key) { DBThreadMessageHdr *threadHdr = GetDBThreadHdrForMsgID(key); MessageKey firstKeyInThread = kIdNone; if (threadHdr == NULL) { //XP_ASSERT(FALSE); (rb) message not found, deleted already but delete key hit too fast for us return firstKeyInThread; } // ### dmb UnreadOnly - this is wrong. firstKeyInThread = threadHdr->GetChildAt(0); delete threadHdr; return firstKeyInThread; } XP_Bool MessageDB::SetPriority(MessageKey key, MSG_PRIORITY priority) { XP_Bool ret; DBMessageHdr *msgHdr = GetDBHdrForKey(key); if (msgHdr == NULL) return FALSE; ret = SetPriority(msgHdr, priority); delete msgHdr; return ret; } XP_Bool MessageDB::SetPriority(DBMessageHdr *msgHdr, MSG_PRIORITY priority) { if ((GetMailDB() != NULL)) { msgHdr->SetPriority(priority); // ###dmb calling SetHdrFlag (on mailDB) this way will basically flush the new mozilla // status which is all we want. Should invent new method. Also, we should only // call this on maildbs, because other db's will leave dirty flag set. SetHdrFlag(msgHdr, TRUE, (MsgFlags) 0); // ###tw When we decide what our priority header and format will be, // this code should also fix the header in the mail msg, // just to be extra paranoid. return TRUE; } else { return FALSE; } } // Helper routine - lowest level of flag setting void MessageDB::SetHdrFlag(DBMessageHdr *msgHdr, XP_Bool bSet, MsgFlags flag) { XP_ASSERT(! (flag & kDirty)); // this won't do the right thing so don't. if (bSet && (!(msgHdr->GetFlags() & flag))) { msgHdr->OrFlags(flag | kDirty); } else if (!bSet && (msgHdr->GetFlags() & flag)) { msgHdr->AndFlags(~flag); msgHdr->OrFlags(kDirty); } } void MessageDB::MarkHdrReadInDB(DBMessageHdr *msgHdr, XP_Bool bRead, ChangeListener *instigator) { SetHdrFlag(msgHdr, bRead, kIsRead); if (m_newSet) m_newSet->Remove(msgHdr->GetMessageKey()); if (m_dbFolderInfo != NULL) { if (bRead) m_dbFolderInfo->ChangeNumNewMessages(-1); else m_dbFolderInfo->ChangeNumNewMessages(1); // DMB TODO m_dbFolderInfo->setDirty(); } NotifyKeyChangeAll(msgHdr->GetMessageKey(), msgHdr->GetFlags(), instigator); } MsgERR MessageDB::MarkRead(MessageKey messageKey, XP_Bool bRead, ChangeListener *instigator) { MsgERR err; DBMessageHdr *msgHdr = GetDBHdrForKey(messageKey); if (msgHdr == NULL) return eID_NOT_FOUND; err = MarkHdrRead(msgHdr, bRead, instigator); delete msgHdr; return err; } MsgERR MessageDB::MarkReplied(MessageKey messageKey, XP_Bool bReplied, ChangeListener *instigator /* = NULL */) { return SetKeyFlag(messageKey, bReplied, kReplied, instigator); } MsgERR MessageDB::MarkForwarded(MessageKey messageKey, XP_Bool bForwarded, ChangeListener *instigator /* = NULL */) { return SetKeyFlag(messageKey, bForwarded, kForwarded, instigator); } MsgERR MessageDB::MarkHasAttachments(MessageKey messageKey, XP_Bool bHasAttachments, ChangeListener *instigator) { return SetKeyFlag(messageKey, bHasAttachments, kHasAttachment, instigator); } MsgERR MessageDB::MarkMarked(MessageKey messageKey, XP_Bool mark, ChangeListener *instigator) { return SetKeyFlag(messageKey, mark, kMsgMarked, instigator); } MsgERR MessageDB::MarkOffline(MessageKey messageKey, XP_Bool offline, ChangeListener *instigator) { return SetKeyFlag(messageKey, offline, kOffline, instigator); } MsgERR MessageDB::MarkImapDeleted(MessageKey messageKey, XP_Bool deleted, ChangeListener *instigator) { return SetKeyFlag(messageKey, deleted, kIMAPdeleted, instigator); } MsgERR MessageDB::MarkMDNNeeded(MessageKey messageKey, XP_Bool bNeeded, ChangeListener *instigator /* = NULL */) { return SetKeyFlag(messageKey, bNeeded, kMDNNeeded, instigator); } MsgERR MessageDB::IsMDNNeeded(MessageKey messageKey, XP_Bool *pNeeded) { MsgERR err = eSUCCESS; DBMessageHdr *msgHdr = GetDBHdrForKey(messageKey); if (msgHdr != NULL && pNeeded) { *pNeeded = ((msgHdr->GetFlags() & kMDNNeeded) == kMDNNeeded); delete msgHdr; return err; } else { return eID_NOT_FOUND; } } MsgERR MessageDB::MarkMDNSent(MessageKey messageKey, XP_Bool bSent, ChangeListener *instigator /* = NULL */) { return SetKeyFlag(messageKey, bSent, kMDNSent, instigator); } MsgERR MessageDB::IsMDNSent(MessageKey messageKey, XP_Bool *pSent) { MsgERR err = eSUCCESS; DBMessageHdr *msgHdr = GetDBHdrForKey(messageKey); if (msgHdr != NULL && pSent) { *pSent = msgHdr->GetFlags() & kMDNSent; delete msgHdr; return err; } else { return eID_NOT_FOUND; } } MsgERR MessageDB::SetKeyFlag(MessageKey messageKey, XP_Bool set, int32 flag, ChangeListener *instigator) { MsgERR err = eSUCCESS; DBMessageHdr *msgHdr = GetDBHdrForKey(messageKey); if (msgHdr == NULL) return eID_NOT_FOUND; SetHdrFlag(msgHdr, set, flag); NotifyKeyChangeAll(msgHdr->GetMessageKey(), msgHdr->GetFlags(), instigator); delete msgHdr; return err; } MsgERR MessageDB::MarkHdrRead(DBMessageHdr *msgHdr, XP_Bool bRead, ChangeListener *instigator) { XP_Bool isRead; IsHeaderRead(msgHdr, &isRead); // if the flag is already correct in the db, don't change it if (!!isRead != !!bRead) { DBThreadMessageHdr *threadHdr = GetDBThreadHdrForMsgID(msgHdr->GetMessageKey()); if (threadHdr != NULL) { threadHdr->MarkChildRead(bRead, m_dbHandle); delete threadHdr; } MarkHdrReadInDB(msgHdr, bRead, instigator); } return eSUCCESS; } MsgERR MessageDB::MarkAllRead(MWContext *context, IDArray *thoseMarked) { MsgERR dbErr; DBMessageHdr *pHeader; ListContext *listContext = NULL; int32 numChanged = 0; char msgBuf[100]; const char * msgTemplate = XP_GetString(MK_MSG_MARKREAD_COUNT); while (TRUE) { dbErr = ListNextUnread(&listContext, &pHeader); if (dbErr == eDBEndOfList) { dbErr = eSUCCESS; break; } // this currently doesn't happen since ListNext doesn't return errors // other than eDBEndOfList. else if (dbErr != eSUCCESS || !pHeader) break; if (numChanged % 10 == 0) { PR_snprintf (msgBuf, sizeof(msgBuf), msgTemplate, numChanged); FE_Progress (context, msgBuf); } if (thoseMarked) thoseMarked->Add(pHeader->GetMessageKey()); dbErr = MarkHdrRead(pHeader, TRUE, NULL); // ### dmb - blow off error? delete pHeader; if (numChanged++ % 200 == 0) // commit every once in a while Commit(); } // force num new to 0. m_dbFolderInfo->ChangeNumNewMessages(-m_dbFolderInfo->GetNumNewMessages()); // DMB TODO m_dbFolderInfo->setDirty(); msgTemplate = XP_GetString(MK_MSG_DONE_MARKREAD_COUNT); PR_snprintf (msgBuf, sizeof(msgBuf), msgTemplate, numChanged); FE_Progress (context, msgBuf); return dbErr; } MsgERR MessageDB::MarkReadByDate (time_t startDate, time_t endDate, MWContext *context, IDArray *markedIds) { MsgERR dbErr; DBMessageHdr *pHeader; ListContext *listContext = NULL; int32 numChanged = 0; char msgBuf[100]; const char * msgTemplate = XP_GetString(MK_MSG_MARKREAD_COUNT); while (TRUE) { if (listContext == NULL) dbErr = ListFirst (&listContext, &pHeader); else dbErr = ListNext(listContext, &pHeader); if (dbErr == eDBEndOfList) { dbErr = eSUCCESS; ListDone(listContext); break; } // this currently doesn't happen since ListNext doesn't return errors // other than eDBEndOfList. else if (dbErr != eSUCCESS) break; time_t headerDate = pHeader->GetDate(); if (headerDate > startDate && headerDate <= endDate) { XP_Bool isRead; IsRead(pHeader->GetMessageKey(), &isRead); if (!isRead) { if (markedIds) markedIds->Add(pHeader->GetMessageKey()); MarkHdrRead(pHeader, TRUE, NULL); // ### dmb - blow off error? if (numChanged % 10 == 0) { PR_snprintf (msgBuf, sizeof(msgBuf), msgTemplate, numChanged); FE_Progress (context, msgBuf); } if (numChanged++ % 1000 == 0) // commit every once in a while Commit(); } } delete pHeader; } msgTemplate = XP_GetString(MK_MSG_DONE_MARKREAD_COUNT); PR_snprintf (msgBuf, sizeof(msgBuf), msgTemplate, numChanged); FE_Progress (context, msgBuf); return dbErr; } MsgERR MessageDB::MarkLater(MessageKey messageKey, time_t until) { XP_ASSERT(m_dbFolderInfo); if (m_dbFolderInfo != NULL) { m_dbFolderInfo->AddLaterKey(messageKey, until); } return eSUCCESS; } void MessageDB::ClearNewList(XP_Bool notify /* = FALSE */) { if (m_newSet) { if (notify) // need to update view { int32 firstMember; while ((firstMember = m_newSet->GetFirstMember()) != 0) { m_newSet->Remove(firstMember); // this bites, since this will cause us to regen new list many times. DBMessageHdr *msgHdr = GetDBHdrForKey(firstMember); if (msgHdr != NULL) { NotifyKeyChangeAll(msgHdr->GetMessageKey(), msgHdr->GetFlags(), NULL); delete msgHdr; } } } delete m_newSet; m_newSet = NULL; } } XP_Bool MessageDB::HasNew() { return m_newSet && m_newSet->getLength() > 0; } MessageKey MessageDB::GetFirstNew() { // even though getLength is supposedly for debugging only, it's the only // way I can tell if the set is empty (as opposed to having a member 0. if (HasNew()) return m_newSet->GetFirstMember(); else return MSG_MESSAGEKEYNONE; } MessageKey MessageDB::GetUnusedFakeId() { ListContext *listContext = NULL; DBMessageHdr *highHdr = NULL; MessageKey fakeMsgKey = kIdStartOfFake; if (ListLast(&listContext, &highHdr) == eSUCCESS) { MessageKey curKey = highHdr->GetMessageKey(); while (curKey == fakeMsgKey || curKey == kIdNone || curKey == kIdPending) { if (curKey == fakeMsgKey) fakeMsgKey--; delete highHdr; highHdr = NULL; if (ListNext(listContext, &highHdr) == eSUCCESS) curKey = highHdr->GetMessageKey(); else break; } if (highHdr) delete highHdr; ListDone(listContext); } return fakeMsgKey; } MsgERR MessageDB::GetUnreadKeyInThread(MessageKey threadId, MessageKey *resultKey, MessageKey *resultThreadId) { MsgERR err = eSUCCESS; DBThreadMessageHdr *threadHdr = GetDBThreadHdrForMsgID(threadId); if (threadHdr == NULL) { #ifdef DEBUG_bienvenu XP_ASSERT(FALSE); #endif return eID_NOT_FOUND; } if (threadHdr->GetNumNewChildren() > 0) { MessageKey startMsg = kIdNone; int numListed; do { const int listChunk = 200; MessageKey listIDs[listChunk]; char listFlags[listChunk]; char listLevels[listChunk]; err = ListIdsInThread(threadHdr->GetThreadID(), &startMsg, listChunk, listIDs, listFlags, listLevels, &numListed); // start at 1, because id 0 is the thread header itself. for (int i = 1; i < numListed; i++) { if (!(listFlags[i] & kIsRead)) { *resultKey = listIDs[i]; if (resultThreadId) *resultThreadId = threadId; break; } } if (numListed < listChunk || startMsg == kIdNone) break; } while (err == eSUCCESS && (*resultKey == kIdNone)); } delete threadHdr; return err; } MsgERR MessageDB::DeleteMessages(IDArray &messageKeys, ChangeListener *instigator) { MsgERR err = eSUCCESS; for (uint index = 0; index < messageKeys.GetSize(); index++) { MessageKey messageKey = messageKeys.GetAt(index); DBMessageHdr *msgHdr = GetDBHdrForKey(messageKey); if (msgHdr == NULL) { err = eID_NOT_FOUND; break; } err = DeleteHeader(msgHdr, instigator, index % 300 == 0); delete msgHdr; if (err != eSUCCESS) break; } Commit(); return err; } XP_Bool MessageDB::AllMessageKeysImapDeleted(const IDArray &messageKeys) { XP_Bool allDeleted = TRUE; for (uint index = 0; allDeleted && (index < messageKeys.GetSize()); index++) { DBMessageHdr *msgHdr = GetDBHdrForKey(messageKeys.GetAt(index)); allDeleted = msgHdr && ((msgHdr->GetFlags() & kIMAPdeleted) != 0); delete msgHdr; } return allDeleted; } MsgERR MessageDB::DeleteMessage(MessageKey messageKey, ChangeListener *instigator, XP_Bool commit) { DBMessageHdr *msgHdr = GetDBHdrForKey(messageKey); if (msgHdr == NULL) return eID_NOT_FOUND; MsgERR ret = DeleteHeader(msgHdr, instigator, commit); delete msgHdr; return ret; } MsgERR MessageDB::DeleteHeader(DBMessageHdr *msgHdr, ChangeListener *instigator, XP_Bool commit, XP_Bool /* onlyRemoveFromThread */) { MessageKey messageKey = msgHdr->GetMessageKey(); // only need to do this for mail - will this speed up news expiration? Is dirtying objects // we're about to delete slow for ? But are we short circuiting some // notifications that we need? // if (GetMailDB()) SetHdrFlag(msgHdr, TRUE, kExpunged); // tell mailbox (mail) if (m_newSet) // if it's in the new set, better get rid of it. m_newSet->Remove(msgHdr->GetMessageKey()); if (m_dbFolderInfo != NULL) { XP_Bool isRead; m_dbFolderInfo->ChangeNumMessages(-1); m_dbFolderInfo->ChangeNumVisibleMessages(-1); IsRead(msgHdr->GetMessageKey(), &isRead); if (!isRead) m_dbFolderInfo->ChangeNumNewMessages(-1); m_dbFolderInfo->m_expunged_bytes += msgHdr->GetByteLength(); // DMB TODO m_dbFolderInfo->setDirty(); } NotifyKeyChangeAll(messageKey, msgHdr->GetFlags(), instigator); // tell listeners // if (!onlyRemoveFromThread) // to speed up expiration, try this. But really need to do this in RemoveHeaderFromDB RemoveHeaderFromDB(msgHdr); if (commit) Commit(); // ### dmb is this a good time to commit? return eSUCCESS; } MsgERR MessageDB::UndoDelete(DBMessageHdr *msgHdr) { MsgERR msgErr = AddHdrToDB(msgHdr, NULL, TRUE); // make sure message is undeleted from source mail folder. // Need to pretend that it's deleted first to reverse it. msgHdr->OrFlags(kExpunged); SetHdrFlag(msgHdr, FALSE, kExpunged); if (m_dbFolderInfo) { m_dbFolderInfo->m_expunged_bytes -= msgHdr->GetByteLength(); // DMB TODO m_dbFolderInfo->setDirty(); } return msgErr; } // This is a lower level routine which doesn't send notifcations or // update folder info. One use is when a rule fires moving a header // from one db to another, to remove it from the first db. void MessageDB::RemoveHeaderFromDB(DBMessageHdr *msgHdr) { // DMB TODO // if (msgHdr->fMark == 0) // msghdr is not in DB! // return; DBThreadMessageHdr *threadHdr = GetDBThreadHdrForMsgID(msgHdr->GetMessageKey()); if (threadHdr != NULL) { threadHdr->RemoveChild(msgHdr->GetMessageKey(), m_dbHandle); // remove empty thread object if it isn't watched or ignored if (threadHdr->GetNumChildren() == 0 && !(threadHdr->GetFlags() & (kWatched | kIgnored))) MSG_DBHandle_RemoveThread(m_dbHandle, threadHdr->GetHandle()); delete threadHdr; } MSG_DBHandle_RemoveHeader(m_dbHandle, msgHdr->GetHandle()); } MsgERR MessageDB::MarkThreadIgnored(DBThreadMessageHdr *threadHdr, MessageKey messageKey, XP_Bool bIgnored, ChangeListener *instigator) { if (bIgnored) { threadHdr->OrFlags(kIgnored); threadHdr->AndFlags(~kWatched); // ignore is implicit un-watch } else threadHdr->AndFlags(~kIgnored); NotifyKeyChangeAll(messageKey, threadHdr->GetFlags(), instigator); return eSUCCESS; } MsgERR MessageDB::MarkThreadWatched(DBThreadMessageHdr *threadHdr, MessageKey messageKey, XP_Bool bWatched, ChangeListener *instigator) { if (bWatched) { threadHdr->AndFlags(~kIgnored); threadHdr->OrFlags(kWatched); } else threadHdr->AndFlags(~kWatched); NotifyKeyChangeAll(messageKey, threadHdr->GetFlags(), instigator); return eSUCCESS; } MsgERR MessageDB::IsMarked(MessageKey messageKey, XP_Bool *pMarked) { DBMessageHdr *msgHdr = GetDBHdrForKey(messageKey); if (msgHdr != NULL) { *pMarked = (msgHdr->GetFlags() & kMsgMarked) != 0; delete msgHdr; return eSUCCESS; } else { return eID_NOT_FOUND; } } MsgERR MessageDB::IsRead(MessageKey messageKey, XP_Bool *pRead) { MsgERR err = eSUCCESS; DBMessageHdr *msgHdr = GetDBHdrForKey(messageKey); if (msgHdr != NULL) { err = IsHeaderRead(msgHdr, pRead); delete msgHdr; return err; } else { return eID_NOT_FOUND; } } uint32 MessageDB::GetStatusFlags(DBMessageHdr *msgHdr) { uint32 statusFlags = msgHdr->GetFlags(); XP_Bool isRead; if (m_newSet && m_newSet->IsMember(msgHdr->GetMessageKey())) statusFlags |= kNew; if (IsRead(msgHdr->GetMessageKey(), &isRead) == eSUCCESS && isRead) statusFlags |= kIsRead; return statusFlags; } MsgERR MessageDB::IsHeaderRead(DBMessageHdr *hdr, XP_Bool *pRead) { if (!hdr) return eID_NOT_FOUND; *pRead = (hdr->GetFlags() & kIsRead) != 0; return eSUCCESS; } MsgERR MessageDB::IsIgnored(MessageKey messageKey, XP_Bool *pIgnored) { XP_ASSERT(pIgnored != NULL); if (!pIgnored) return eBAD_PARAMETER; DBThreadMessageHdr *threadHdr = GetDBThreadHdrForMsgID(messageKey); // This should be very surprising, but we leave that up to the caller // to determine for now. if (threadHdr == NULL) return eID_NOT_FOUND; *pIgnored = (threadHdr->GetFlags() & kIgnored) ? TRUE : FALSE; delete threadHdr; return eSUCCESS; } MsgERR MessageDB::HasAttachments(MessageKey messageKey, XP_Bool *pHasThem) { XP_ASSERT(pHasThem != NULL); if (!pHasThem) return eBAD_PARAMETER; DBThreadMessageHdr *threadHdr = GetDBThreadHdrForMsgID(messageKey); // This should be very surprising, but we leave that up to the caller // to determine for now. if (threadHdr == NULL) return eID_NOT_FOUND; *pHasThem = (threadHdr->GetFlags() & kHasAttachment) ? TRUE : FALSE; delete threadHdr; return eSUCCESS; } // This function goes through the list of latered documents and marks them // unread if the current date/time is > than the latered "until" setting. // Since that is currently always 0, this routine should mark everything // in the latered list unread. void MessageDB::HandleLatered() { time_t curTime = XP_TIME(); if (!m_dbFolderInfo) return; for (int32 laterIndex = 0; laterIndex < m_dbFolderInfo->GetNumLatered(); ) { time_t until; MessageKey laterKey = m_dbFolderInfo->GetLateredAt(laterIndex, &until); if (curTime > until) { MarkRead(laterKey, FALSE, NULL); m_dbFolderInfo->RemoveLateredAt(laterIndex); } else { laterIndex++; } } } void MessageDB::SetSortInfo(SortType sortType, SortOrder sortOrder) { if (m_dbFolderInfo) m_dbFolderInfo->SetSortInfo(sortType, sortOrder); } MsgERR MessageDB::GetSortInfo(SortType *pSortType, SortOrder *pSortOrder) { if (!(pSortType && pSortOrder && m_dbFolderInfo)) { XP_ASSERT(FALSE); return eBAD_PARAMETER; } m_dbFolderInfo->GetSortInfo(pSortType, pSortOrder); return eSUCCESS; } // Get a handle for a document given its message number. Because a subclass // of MsgDocument may be returned, we need to return a pointer to an allocated object. // For some reason, the UI has decided to force a purge of the database. MsgERR MessageDB::Purge() { FinishAddingHeaders(); // this will add m_newHeaders to the db return eSUCCESS; } //----------------------------------------------------------------------------- // EnableCache //----------------------------------------------------------------------------- XP_Bool MessageDB::EnableCache(XP_Bool enable) { XP_Bool oldVal = m_cacheEnabled; m_cacheEnabled = enable; return(oldVal); } //---------------------------------------------------------------------- // FindInCache //---------------------------------------------------------------------- MessageDB* MessageDB::FindInCache(const char * pDbName) { for (int i = 0; i < GetDBCache()->GetSize(); i++) { MessageDB* pMessageDB = GetDBCache()->GetAt(i); if (pMessageDB->MatchDbName(pDbName)) { return(pMessageDB); } } return(NULL); } //---------------------------------------------------------------------- // FindInCache //---------------------------------------------------------------------- int MessageDB::FindInCache(MessageDB* pMessageDB) { for (int i = 0; i < GetDBCache()->GetSize(); i++) { if (GetDBCache()->GetAt(i) == pMessageDB) { return(i); } } return(-1); } //---------------------------------------------------------------------- // RemoveFromCache //---------------------------------------------------------------------- void MessageDB::RemoveFromCache(MessageDB* pMessageDB) { int i = FindInCache(pMessageDB); if (i != -1) { GetDBCache()->RemoveAt(i); } } #ifdef DEBUG void MessageDB::DumpCache() { for (int i = 0; i < GetDBCache()->GetSize(); i++) { #ifdef DEBUG_bienvenu MessageDB* pMessageDB = #endif GetDBCache()->GetAt(i); #ifdef DEBUG_bienvenu XP_Trace("db %s in cache use count = %d\n", pMessageDB->m_dbName, pMessageDB->m_useCount); #endif } } #endif XP_Bool MessageDB::MatchDbName(const char * dbName) // returns TRUE if they match { XP_ASSERT(m_dbName); return !XP_FILENAMECMP(dbName, m_dbName); } MsgERR MessageDB::FinishAddingHeaders() { MsgERR err = eSUCCESS; XP_Bool isNewThread; // go through the new headers adding them to the db // The idea here is that m_headers is just the new headers for (int i = 0; i < m_newHeaders.GetSize(); i++) { DBMessageHdr *dbMsgHdr = (DBMessageHdr *) m_newHeaders[i]; err = AddHdrToDB(dbMsgHdr, &isNewThread); delete dbMsgHdr; } m_headerIndex = 0; m_newHeaders.RemoveAll(); return err; // m_headerIndex = 0; // return eSUCCESS; } DBThreadMessageHdr *MessageDB::GetDBThreadHdrForSubject(DBMessageHdr *msgHdr) { MSG_ThreadHandle threadHandle = MSG_DBHandle_GetThreadHandleForMsgHdrSubject(m_dbHandle, msgHdr->GetHandle()); return GetThreadHeaderFromHandle(threadHandle); } // This functions takes a string message id and returns the // corresponding message hdr DBThreadMessageHdr *MessageDB::GetDBMsgHdrForReference(const char * msgID) { DBMessageHdr *headerObject = GetDBMessageHdrForID(msgID); DBThreadMessageHdr *thread = NULL; if (headerObject != NULL) { // find thread header for header whose message id we matched. thread = GetDBThreadHdrForMsgID(headerObject->GetMessageKey()); delete headerObject; } return thread; } // make the passed in header a thread header MsgERR MessageDB::AddThread(DBMessageHdr *msgHdr) { //TRACE("entering AddThread\n"); MSG_ThreadHandle threadHandle = MSG_DBHandle_AddThreadFromMsgHandle(m_dbHandle, msgHdr->GetHandle()); DBThreadMessageHdr *threadHdr = new DBThreadMessageHdr(threadHandle); AddToThread(msgHdr, threadHdr, FALSE); XP_ASSERT(threadHdr->GetThreadID() == msgHdr->GetThreadId()); delete threadHdr; // If this header has references, we might want to create an expired // header for this thread, instead of promoting this header to thread status. // In particular, the real thread header might arrive later, in which case // we would just turn off the expired bit on the dummy. Of course, it // could be anyone of the parents, and is more likely to be an immediate ancestor... // which would argue for making the message-id be the last reference, not the first. // But for now, just make it a top-level thread - we can always rearrange things // if a better top-level thread header comes in. Or if we decide to have a dummy // header... // AddHdr(msgHdr); //TRACE("adding thread %s\n", (const char *) dummyHdr->m_subject); #ifdef _DEBUG1 // check that we can pull it out of the database. DBThreadMessageHdr *newHdr; newHdr = GetDBMsgHdrForReference(dummyHdr->m_messageId); ASSERT(newHdr != NULL); delete newHdr; #endif //TRACE("leaving AddThread\n"); return eSUCCESS; } // really add it to DB. MsgERR MessageDB::AddHdr(DBMessageHdr *hdr) { // TODO - need to do exception handling. MSG_DBHandle_AddHeader(m_dbHandle, hdr->GetHandle()); return eSUCCESS; } void MessageDB::AddToNewList(MessageKey key) { if (m_newSet == NULL) m_newSet = msg_NewsArtSet::Create(); if (m_newSet) m_newSet->Add(key); } // add a header to the database, and thread it. // For now, it's OK if newThread or resultHdr are NULL MsgERR MessageDB::AddHdrToDB(DBMessageHdr *newHdr, XP_Bool *newThread, XP_Bool notify /* = FALSE */) { MsgERR err = eSUCCESS; DBThreadMessageHdr *refHdr = NULL; if (m_addCount >= m_commitChunk) { Commit(); m_addCount = 0; } if (newHdr == NULL) return err; #define SUBJ_THREADING 1// try reference threading first for (int32 i = 0; i < newHdr->GetNumReferences(); i++) { MSG_ThreadHandle refHdrThreadHandle = MSG_HeaderHandle_GetThreadForReference(newHdr->GetHandle(), i, m_dbHandle); if (refHdrThreadHandle) { refHdr = GetThreadHeaderFromHandle(refHdrThreadHandle); if (refHdr) { newHdr->SetThreadId(refHdr->GetThreadID()); err = AddToThread(newHdr, refHdr, TRUE); } break; } } #ifdef SUBJ_THREADING // try subject threading if we couldn't find a reference and the subject starts with Re: if ((newHdr->GetFlags() & kHasRe) && refHdr == NULL && (refHdr = GetDBThreadHdrForSubject(newHdr)) != NULL) { newHdr->SetThreadId(refHdr->GetThreadID()); //TRACE("threading based on subject %s\n", (const char *) msgHdr->m_subject); // AddHdr(newHdr); // if we move this and do subject threading after, ref threading, // don't thread within children, since we know it won't work. But for now, pass TRUE. err = AddToThread(newHdr, refHdr, TRUE); } #endif // SUBJ_THREADING XP_ASSERT( newHdr != NULL); if (refHdr == NULL) { // couldn't find any parent articles - msgHdr is top-level thread, for now err = AddThread(newHdr); if (newThread) *newThread = TRUE; } else { if (newThread) *newThread = FALSE; } delete refHdr; // update if (err == eSUCCESS) { if ((newHdr->GetFlags() & kNew)) { newHdr->AndFlags(~kNew); // make sure not filed out AddToNewList(newHdr->GetMessageKey()); } if (m_dbFolderInfo != NULL) { m_dbFolderInfo->ChangeNumMessages(1); m_dbFolderInfo->ChangeNumVisibleMessages(1); if (! (newHdr->GetFlags() & kIsRead)) m_dbFolderInfo->ChangeNumNewMessages(1); // dmb todo m_dbFolderInfo->setDirty(); } if (notify) NotifyKeyChangeAll(newHdr->GetMessageKey(), newHdr->GetFlags() | kAdded, NULL); } m_addCount++; return err; } MsgERR MessageDB::AddToThread(DBMessageHdr *reply, DBThreadMessageHdr *threadHdr, XP_Bool threadInThread) { reply->SetLevel(0); // for now, until we get threading within a thread. reply->SetThreadId(threadHdr->GetThreadID()); // determine where to add to thread. threadHdr->AddChild(reply, this, threadInThread); return eSUCCESS; } // Get the header for the passed in message-id. Could return a DBMailMessageHdr // because we do a deep find. Caller must RemoveReference DBMessageHdr when done with it. DBMessageHdr *MessageDB::GetDBMessageHdrForID(const char * msgID) { DBMessageHdr *headerObject = NULL; MSG_HeaderHandle headerHandle = MSG_DBHandle_GetHandleForMessageID(m_dbHandle, msgID); if (headerHandle) GetHeaderFromHandle(headerHandle, &headerObject); return headerObject; } MessageKey MessageDB::GetMessageKeyForID(const char *msgID) { MessageKey retKey = kIdNone; DBMessageHdr *msgHdr = GetDBMessageHdrForID(msgID); if (msgHdr) { retKey = msgHdr->GetMessageKey(); delete msgHdr; } return retKey; } // caller needs to RemoveReference when finished. DBThreadMessageHdr *MessageDB::GetDBThreadHdrForThreadID(MessageKey messageKey) { MSG_ThreadHandle threadHandle = MSG_DBHandle_GetThreadHeaderForThreadID(m_dbHandle, messageKey); return GetThreadHeaderFromHandle(threadHandle); } DBThreadMessageHdr *MessageDB::GetDBThreadHdrForMsgHdr(DBMessageHdr *msgHdr) { DBThreadMessageHdr *threadHdr = GetDBThreadHdrForThreadID(msgHdr->GetThreadId()); return threadHdr; } // Given the id of a message, find the thread header for the message's thread // Returns NULL if we can't find the message hdr, or its thread. // Caller needs to RemoveReference thread header DBThreadMessageHdr *MessageDB::GetDBThreadHdrForMsgID(MessageKey messageKey) { DBMessageHdr *msgHdr = GetDBHdrForKey(messageKey); if (msgHdr == NULL) return NULL; DBThreadMessageHdr *threadHdr = GetDBThreadHdrForMsgHdr(msgHdr); delete msgHdr; return threadHdr; } // Given a MessageKey, return the threadId of its thread, or kIdNone // if we can't find the given MessageKey. MessageKey MessageDB::GetThreadIdForMsgId(MessageKey messageKey) { DBMessageHdr *msgHdr = GetDBHdrForKey(messageKey); if (msgHdr != NULL) { MessageKey threadId = msgHdr->GetThreadId(); delete msgHdr; return threadId; } else { return kIdNone; } } // caller needs to delete when finished. DBMessageHdr *MessageDB::GetDBHdrForKey(MessageKey messageKey) { MSG_DBHandle dbHandle = MSG_DBHandle_GetHandleForKey(m_dbHandle, messageKey); DBMessageHdr *headerObject = NULL; if (dbHandle) headerObject = new DBMessageHdr(dbHandle); return headerObject; } // Test if the key we're about to add already exists (in which case // the caller shouldn't add. This can happen in news for various reasons // and should be handled). XP_Bool MessageDB::KeyToAddExists(MessageKey messageKey) { // this relies on GetMessageHdr not touching msgHdr if it doesn't find key in db DBMessageHdr *msgHdr = GetDBHdrForKey(messageKey); if (msgHdr != NULL) { // this handles the bizarre case of the the db having all the // headers but not having the highwater mark set, in which // case it will always retrieve all headers. if (m_dbFolderInfo != NULL) m_dbFolderInfo->SetHighWater(messageKey); delete msgHdr; return TRUE; } return FALSE; } ///////////////////// MessageHdrStruct methods void MessageHdrStruct::SetSubject(const char * subject) { if (msg_StripRE(&subject, NULL)) { m_flags |= kHasRe; } else { m_flags &= ~kHasRe; } XP_STRNCPY_SAFE(m_subject, subject, sizeof(m_subject)); } void MessageHdrStruct::SetAuthor(const char * author) { XP_STRNCPY_SAFE(m_author, author, sizeof(m_author)); } // Set message id, stripping off leading '<' and trailing '>', if any void MessageHdrStruct::SetMessageID(const char * msgID) { if (msgID) StripMessageId(msgID, m_messageId, sizeof(m_messageId)); } /* static */void MessageHdrStruct::StripMessageId(const char *msgID, char *outMsgId, int msgIdLen) { if (*msgID == '<') msgID++; XP_STRNCPY_SAFE(outMsgId, msgID, msgIdLen); char * lastChar = outMsgId + strlen(outMsgId) -1; if (*lastChar == '>') *lastChar = '\0'; } void MessageHdrStruct::SetReferences(const char * referencesStr) { if (referencesStr) XP_STRNCPY_SAFE(m_references, referencesStr, sizeof(m_references)); } void MessageHdrStruct::SetDate(const char * date) { m_date = XP_ParseTimeString (date, FALSE); } void MessageHdrStruct::SetLines(uint32 lines) { m_messageSize = lines; } void MessageHdrStruct::SetSize(uint32 size) { m_messageSize = size; } // get the next <> delimited reference from nextRef and copy it into reference, // which is a pointer to a buffer at least kMaxMsgIdLen long. const char * MessageHdrStruct::GetReference(const char *nextRef, char *reference) { const char *ptr = nextRef; while ((*ptr == '<' || *ptr == ' ') && *ptr) ptr++; for (int i = 0; *ptr && *ptr != '>' && i < kMaxMsgIdLen; i++) *reference++ = *ptr++; if (*ptr == '>') ptr++; *reference = '\0'; return ptr; } // Copy the corresponding fields from a full message header into a short message hdr. void MessageDB::CopyFullHdrToShortHdr(MSG_MessageLine *msgHdr, MessageHdrStruct *fullHdr) { msgHdr->threadId = fullHdr->m_threadId; msgHdr->messageKey = fullHdr->m_messageKey; //for threads, same as threadId XP_STRNCPY_SAFE(msgHdr->subject, fullHdr->m_subject, sizeof(msgHdr->subject)); XP_STRNCPY_SAFE(msgHdr->author, fullHdr->m_author, sizeof(msgHdr->author)); msgHdr->date = fullHdr->m_date; msgHdr->messageLines = fullHdr->m_messageSize; // lines for news articles, // bytes for mail messages // ###tw Is the above true // yet? msgHdr->priority = fullHdr->m_priority; msgHdr->flags = fullHdr->m_flags; msgHdr->level = fullHdr->m_level; // indentation level msgHdr->numChildren = fullHdr->m_numChildren; // for top-level threads msgHdr->numNewChildren = fullHdr->m_numNewChildren; // for top-level threads } // static helper functions to convert between kFlags and MSG_FLAG_* void MessageDB::ConvertDBFlagsToPublicFlags(uint32 *flags) { uint32 publicFlags = 0; publicFlags = (kSameAsMSG_FLAG & *flags); if (*flags & kExpunged) // is this needed? publicFlags |= MSG_FLAG_EXPUNGED; if (*flags & kHasRe) publicFlags |= MSG_FLAG_HAS_RE; if (*flags & kIgnored) publicFlags |= MSG_FLAG_IGNORED; if (*flags & kPartial) publicFlags |= MSG_FLAG_PARTIAL; if (*flags & kMDNNeeded) publicFlags |= MSG_FLAG_MDN_REPORT_NEEDED; if (*flags & kMDNSent) publicFlags |= MSG_FLAG_MDN_REPORT_SENT; if (*flags & kTemplate) publicFlags |= MSG_FLAG_TEMPLATE; *flags = publicFlags; } void MessageDB::ConvertPublicFlagsToDBFlags(uint32 *flags) { uint32 dbFlags = 0; dbFlags = (kSameAsMSG_FLAG & *flags); if (*flags & MSG_FLAG_EXPUNGED) // is this needed? dbFlags |= kExpunged ; if (*flags & MSG_FLAG_HAS_RE) dbFlags |= kHasRe ; if (*flags & MSG_FLAG_IGNORED) dbFlags |= kIgnored; if (*flags & MSG_FLAG_PARTIAL) dbFlags |= kPartial; if (*flags & MSG_FLAG_MDN_REPORT_NEEDED) dbFlags |= kMDNNeeded; if (*flags & MSG_FLAG_MDN_REPORT_SENT) dbFlags |= kMDNSent; if (*flags & MSG_FLAG_TEMPLATE) dbFlags |= kTemplate; *flags = dbFlags; } ViewType MessageDB::GetViewType() { ViewType retViewType = ViewAllThreads; if (m_dbFolderInfo) { retViewType = (ViewType) m_dbFolderInfo->GetViewType(); if (retViewType == ViewKilledThreads) { retViewType = ViewAllThreads; m_dbFolderInfo->SetFlags(m_dbFolderInfo->GetFlags() | MSG_FOLDER_PREF_SHOWIGNORED); } } return retViewType; } void MessageDB::SetViewType(ViewType viewType) { if (m_dbFolderInfo) m_dbFolderInfo->SetViewType(viewType); else XP_ASSERT(FALSE); } MsgERR MessageDB::GetCachedPassword(XPStringObj &cachedPassword) { m_dbFolderInfo->GetCachedPassword(cachedPassword, m_dbHandle); return eSUCCESS; } MsgERR MessageDB::SetCachedPassword(const char *password) { m_dbFolderInfo->SetCachedPassword(password, m_dbHandle); return eSUCCESS; } XP_Bool MessageDB::HasCachedPassword() { XPStringObj password; m_dbFolderInfo->GetCachedPassword(password, m_dbHandle); return (XP_STRLEN(password) > 0); }