gecko-dev/lib/libmsg/newshost.cpp

2213 lines
55 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):
*/
#include "rosetta.h"
#include "msg.h"
#include "newshost.h"
#include "newsset.h"
#include "msgfinfo.h"
#include "msgfpane.h"
#include "hosttbl.h"
#include "grec.h"
#include "prefapi.h"
#ifdef XP_UNIX
static const char LINEBREAK_START = '\012';
#else
static const char LINEBREAK_START = '\015';
#endif
MSG_NewsHost* MSG_NewsHost::M_FileOwner = NULL;
extern "C" {
extern int MK_OUT_OF_MEMORY;
extern int MK_UNABLE_TO_OPEN_NEWSRC;
extern int MK_MIME_ERROR_WRITING_FILE;
HG28366
extern int MK_MSG_CANT_MOVE_FOLDER;
}
MSG_NewsHost::MSG_NewsHost(MSG_Master* master, const char* name,
XP_Bool xxx, int32 port)
{
m_master = master;
m_hostname = new char [XP_STRLEN(name) + 1];
XP_STRCPY(m_hostname, name);
HG18763
XP_ASSERT(port);
if (port == 0) port = HG28736 NEWS_PORT;
m_port = port;
m_searchableGroupCharsets = XP_HashTableNew(20, XP_StringHash,
(XP_HashCompFunction) strcmp);
m_nameAndPort = NULL;
m_fullUIName = NULL;
m_optionLines = NULL;
m_filename = NULL;
m_groups = NULL;
m_dbfilename = NULL;
m_dirty = 0;
m_writetimer = NULL;
m_postingAllowed = TRUE;
m_supportsExtensions = FALSE;
m_pushAuth = FALSE;
m_groupSucceeded = FALSE;
}
MSG_NewsHost::~MSG_NewsHost()
{
if (m_dirty) WriteNewsrc();
if (m_groupTreeDirty) SaveHostInfo();
FREEIF(m_optionLines);
delete [] m_filename;
delete [] m_hostname;
FREEIF(m_nameAndPort);
FREEIF(m_fullUIName);
delete m_groups;
delete [] m_dbfilename;
delete m_groupTree;
if (m_block)
delete [] m_block;
if (m_groupFile)
XP_FileClose (m_groupFile);
FREEIF(m_groupFilePermissions);
if (M_FileOwner == this) M_FileOwner = NULL;
FREEIF(m_hostinfofilename);
int i;
for (i = 0; i < m_supportedExtensions.GetSize(); i++)
XP_FREE((char*) m_supportedExtensions.GetAt(i));
for (i = 0; i < m_searchableGroups.GetSize(); i++)
XP_FREE((char*) m_searchableGroups.GetAt(i));
for (i = 0; i < m_searchableHeaders.GetSize(); i++)
XP_FREE((char*) m_searchableHeaders.GetAt(i));
for (i = 0; i < m_propertiesForGet.GetSize(); i++)
XP_FREE((char*) m_propertiesForGet.GetAt(i));
for (i = 0; i < m_valuesForGet.GetSize(); i++)
XP_FREE((char*) m_valuesForGet.GetAt(i));
if (m_searchableGroupCharsets)
{
// We do NOT free the individual key/value pairs,
// because deleting m_searchableGroups above has
// already caused this to happen.
XP_HashTableDestroy(m_searchableGroupCharsets);
m_searchableGroupCharsets = NULL;
}
}
void
MSG_NewsHost::OpenGroupFile(const XP_FilePerm permissions)
{
XP_ASSERT(permissions);
if (!permissions) return;
if (m_groupFile) {
if (m_groupFilePermissions &&
XP_STRCMP(m_groupFilePermissions, permissions) == 0) {
return;
}
XP_FileClose(m_groupFile);
m_groupFile = NULL;
}
if (M_FileOwner && M_FileOwner != this && M_FileOwner->m_groupFile) {
XP_FileClose(M_FileOwner->m_groupFile);
M_FileOwner->m_groupFile = NULL;
}
M_FileOwner = this;
FREEIF(m_groupFilePermissions);
m_groupFilePermissions = XP_STRDUP(permissions);
m_groupFile = XP_FileOpen(m_hostinfofilename, xpXoverCache, permissions);
}
void MSG_NewsHost::ClearNew()
{
m_firstnewdate = time(0) + 1;
}
void MSG_NewsHost::dump()
{
// ###tw Write me...
}
int32 MSG_NewsHost::getPort()
{
return m_port;
}
HG28361
time_t MSG_NewsHost::getLastUpdate()
{
return m_lastgroupdate;
}
void MSG_NewsHost::setLastUpdate(time_t date)
{
m_lastgroupdate = date;
}
const char* MSG_NewsHost::getStr()
{
return m_hostname;
}
const char* MSG_NewsHost::getNameAndPort()
{
if (!m_nameAndPort) {
if (m_port != (HG23837 NEWS_PORT)) {
m_nameAndPort = PR_smprintf("%s:%ld", m_hostname, long(m_port));
} else {
m_nameAndPort = XP_STRDUP(m_hostname);
}
}
return m_nameAndPort;
}
const char* MSG_NewsHost::getFullUIName()
{
if (!m_fullUIName) {
if (HG65276) return getNameAndPort();
m_fullUIName = PR_smprintf("%s%s", getNameAndPort(),
HG38467);
}
return m_fullUIName;
}
int32
MSG_NewsHost::RememberLine(char* line)
{
char* new_data;
if (m_optionLines) {
new_data =
(char *) XP_REALLOC(m_optionLines,
XP_STRLEN(m_optionLines)
+ XP_STRLEN(line) + 4);
} else {
new_data = (char *) XP_ALLOC(XP_STRLEN(line) + 3);
}
if (!new_data) return MK_OUT_OF_MEMORY;
XP_STRCPY(new_data, line);
XP_STRCAT(new_data, LINEBREAK);
m_optionLines = new_data;
return 0;
}
int32
MSG_NewsHost::ProcessLine_s(char* line, uint32 line_size, void* closure)
{
return ((MSG_NewsHost*) closure)->ProcessLine(line, line_size);
}
int32
MSG_NewsHost::ProcessLine(char* line, uint32 line_size)
{
/* guard against blank line lossage */
if (line[0] == '#' || line[0] == CR || line[0] == LF) return 0;
line[line_size] = 0;
if ((line[0] == 'o' || line[0] == 'O') &&
!strncasecomp (line, "options", 7)) {
return RememberLine(line);
}
char *s;
char *end = line + line_size;
static msg_NewsArtSet *set;
for (s = line; s < end; s++)
if (*s == ':' || *s == '!')
break;
if (*s == 0) {
/* What is this? Well, don't just throw it away... */
return RememberLine(line);
}
set = msg_NewsArtSet::Create(s + 1, this);
if (!set) return MK_OUT_OF_MEMORY;
XP_Bool subscribed = (*s == ':');
*s = '\0';
if (strlen(line) == 0)
{
delete set;
return 0;
}
MSG_FolderInfoNews* info;
if (subscribed && IsCategoryContainer(line))
{
info = new MSG_FolderInfoCategoryContainer(line, set, subscribed,
this,
m_hostinfo->GetDepth() + 1);
msg_GroupRecord* group = FindOrCreateGroup(line);
// Go add all of our categories to the newsrc.
AssureAllDescendentsLoaded(group);
msg_GroupRecord* end = group->GetSiblingOrAncestorSibling();
msg_GroupRecord* child;
for (child = group->GetNextAlphabetic() ;
child != end ;
child = child->GetNextAlphabetic()) {
XP_ASSERT(child);
if (!child) break;
char* fullname = child->GetFullName();
if (!fullname) break;
MSG_FolderInfoNews* info = FindGroup(fullname);
if (!info) { // autosubscribe, if we haven't seen this one.
char* groupLine = PR_smprintf("%s:", fullname);
if (groupLine) {
ProcessLine(groupLine, XP_STRLEN(groupLine));
XP_FREE(groupLine);
}
}
delete [] fullname;
}
}
else
info = new MSG_FolderInfoNews(line, set, subscribed, this,
m_hostinfo->GetDepth() + 1);
if (!info) return MK_OUT_OF_MEMORY;
// for now, you can't subscribe to category by itself.
if (! info->IsCategory())
{
XPPtrArray* infolist = (XPPtrArray*) m_hostinfo->GetSubFolders();
infolist->Add(info);
}
m_groups->Add(info);
// prime the folder info from the folder cache while it's still around.
// Except this might disable the update of new counts - check it out...
m_master->InitFolderFromCache (info);
return 0;
}
int MSG_NewsHost::LoadNewsrc(MSG_FolderInfo* hostinfo)
{
char *ibuffer = 0;
uint32 ibuffer_size = 0;
uint32 ibuffer_fp = 0;
m_hostinfo = hostinfo;
XP_ASSERT(m_hostinfo);
if (!m_hostinfo) return -1;
int status = 0;
FREEIF(m_optionLines);
if (!m_groups) {
int size = 2048;
char* buffer;
buffer = new char[size];
if (!buffer) return MK_OUT_OF_MEMORY;
m_groups = new MSG_FolderArray();
if (!m_groups) {
delete [] buffer;
return MK_OUT_OF_MEMORY;
}
XP_File fid = XP_FileOpen(GetNewsrcFileName(),
HG84636 xpNewsRC,
XP_FILE_READ_BIN);
if (fid) {
do {
status = XP_FileRead(buffer, size, fid);
if (status > 0) {
msg_LineBuffer(buffer, status,
&ibuffer, &ibuffer_size, &ibuffer_fp,
FALSE,
#ifdef XP_OS2
(int32 (_Optlink*) (char*,uint32,void*))
#endif
MSG_NewsHost::ProcessLine_s, this);
}
} while (status > 0);
XP_FileClose(fid);
}
if (status == 0 && ibuffer_fp > 0) {
status = ProcessLine_s(ibuffer, ibuffer_fp, this);
ibuffer_fp = 0;
}
delete [] buffer;
FREEIF(ibuffer);
}
// build up the category tree for each category container so that roll-up
// of counts will work before category containers are opened.
for (int32 i = 0; i < m_groups->GetSize(); i++) {
MSG_FolderInfoNews *info = (MSG_FolderInfoNews *) m_groups->GetAt(i);
if (info->GetType() == FOLDER_CATEGORYCONTAINER) {
const char* groupname = info->GetNewsgroupName();
msg_GroupRecord* group = m_groupTree->FindDescendant(groupname);
XP_ASSERT(group);
if (group) {
MSG_FolderInfoCategoryContainer *catContainer =
(MSG_FolderInfoCategoryContainer *) info;
info->SetFlag(MSG_FOLDER_FLAG_ELIDED | MSG_FOLDER_FLAG_DIRECTORY);
catContainer->BuildCategoryTree(catContainer, groupname, group,
2, m_master);
}
}
}
return 0;
}
int
MSG_NewsHost::WriteNewsrc()
{
XP_ASSERT(m_groups);
if (!m_groups) return -1;
XP_ASSERT(m_dirty);
// Just to be sure. It's safest to go ahead and write it out anyway,
// even if we do somehow get called without the dirty bit set.
XP_File fid = XP_FileOpen(GetNewsrcFileName(), xpTemporaryNewsRC,
XP_FILE_WRITE_BIN);
if (!fid) return MK_UNABLE_TO_OPEN_NEWSRC;
int status = 0;
#ifdef XP_UNIX
/* Clone the permissions of the "real" newsrc file into the temp file,
so that when we rename the finished temp file to the real file, the
preferences don't appear to change. */
{
XP_StatStruct st;
if (XP_Stat (GetNewsrcFileName(), &st,
HG48362 xpNewsRC) == 0)
/* Ignore errors; if it fails, bummer. */
/* SCO doesn't define fchmod at all. no big deal.
AIX3, however, defines it to take a char* as the first parameter. The
man page sez it takes an int, though. ... */
#if defined( SCO_SV ) || defined ( AIXV3 )
{
char *really_tmp_file = WH_FileName(GetNewsrcFileName(), xpTemporaryNewsRC);
chmod (really_tmp_file, (st.st_mode | S_IRUSR | S_IWUSR));
XP_FREE(really_tmp_file);
}
#else
fchmod (fileno(fid), (st.st_mode | S_IRUSR | S_IWUSR));
#endif
}
#endif /* XP_UNIX */
if (m_optionLines) {
status = XP_FileWrite(m_optionLines, XP_STRLEN(m_optionLines), fid);
if (status < int(XP_STRLEN(m_optionLines))) {
status = MK_MIME_ERROR_WRITING_FILE;
}
}
int n = m_groups->GetSize();
for (int i=0 ; i<n && status >= 0 ; i++) {
MSG_FolderInfoNews* info = (MSG_FolderInfoNews*) ((*m_groups)[i]);
// GetNewsFolderInfo will get root category for cat container.
char* str = info->GetNewsFolderInfo()->GetSet()->Output();
if (!str) {
status = MK_OUT_OF_MEMORY;
break;
}
char* line = PR_smprintf("%s%s %s" LINEBREAK, info->GetNewsgroupName(),
info->IsSubscribed() ? ":" : "!", str);
if (!line) {
delete [] str;
status = MK_OUT_OF_MEMORY;
break;
}
int length = XP_STRLEN(line);
status = XP_FileWrite(line, length, fid);
if (status < length) {
status = MK_MIME_ERROR_WRITING_FILE;
}
delete [] str;
XP_FREE(line);
}
XP_FileClose(fid);
if (status >= 0) {
if (XP_FileRename(GetNewsrcFileName(), xpTemporaryNewsRC,
GetNewsrcFileName(),
HG72626 xpNewsRC) < 0) {
status = MK_MIME_ERROR_WRITING_FILE;
}
}
if (status < 0) {
XP_FileRemove(GetNewsrcFileName(), xpTemporaryNewsRC);
return status;
}
m_dirty = FALSE;
if (m_writetimer) {
FE_ClearTimeout(m_writetimer);
m_writetimer = NULL;
}
if (m_groupTreeDirty) SaveHostInfo();
return status;
}
int
MSG_NewsHost::WriteIfDirty()
{
if (m_dirty) return WriteNewsrc();
return 0;
}
void
MSG_NewsHost::MarkDirty()
{
m_dirty = TRUE;
if (!m_writetimer)
m_writetimer = FE_SetTimeout((TimeoutCallbackFunction)MSG_NewsHost::WriteTimer, this,
5L * 60L * 1000L); // Hard-coded -- 5 minutes.
}
void
MSG_NewsHost::WriteTimer(void* closure)
{
MSG_NewsHost* host = (MSG_NewsHost*) closure;
host->m_writetimer = NULL;
if (host->WriteNewsrc() < 0) {
// ###tw Pop up error message???
host->MarkDirty(); // Cause us to try again. Or is this bad? ###tw
}
}
const char*
MSG_NewsHost::GetNewsrcFileName()
{
return m_filename;
}
int
MSG_NewsHost::SetNewsrcFileName(const char* name)
{
delete [] m_filename;
m_filename = new char [XP_STRLEN(name) + 1];
if (!m_filename) return MK_OUT_OF_MEMORY;
XP_STRCPY(m_filename, name);
m_hostinfofilename = PR_smprintf("%s/hostinfo.dat", GetDBDirName());
if (!m_hostinfofilename) return MK_OUT_OF_MEMORY;
XP_StatStruct st;
if (XP_Stat (m_hostinfofilename, &st, xpXoverCache) == 0) {
m_fileSize = st.st_size;
}
if (m_groupFile) {
XP_FileClose(m_groupFile);
m_groupFile = NULL;
}
OpenGroupFile();
if (!m_groupFile) {
// It does not exist. Create an empty one. But first, we go through
// the whole directory and blow away anything that does not end in
// ".snm". This is to take care of people running B1 code, which
// kept databases with such filenames.
int lastcount = -1;
int count = -1;
do {
XP_Dir dir = XP_OpenDir(GetDBDirName(), xpXoverCache);
if (!dir) break;
lastcount = count;
count = 0;
XP_DirEntryStruct *entry = NULL;
for (entry = XP_ReadDir(dir); entry; entry = XP_ReadDir(dir)) {
count++;
const char* name = entry->d_name;
int length = XP_STRLEN(name);
if (name[0] == '.' || name[0] == '#' ||
name[length - 1] == '~') continue;
if (length < 4 ||
XP_STRCASECMP(name + length - 4, ".snm") != 0) {
char* tmp = PR_smprintf("%s/%s", GetDBDirName(), name);
if (tmp) {
XP_FileRemove(tmp, xpXoverCache);
XP_FREE(tmp);
}
}
}
XP_CloseDir(dir);
} while (lastcount != count); // Keep doing it over and over, until
// we get the same number of entries in
// the dir. This is because I do not
// trust XP_ReadDir() to do the right
// thing if I delete files out from
// under it.
// OK, now create the new empty hostinfo file.
OpenGroupFile(XP_FILE_WRITE_BIN);
XP_ASSERT(m_groupFile);
if (!m_groupFile)
return -1;
OpenGroupFile();
XP_ASSERT(m_groupFile);
if (!m_groupFile)
return -1;
m_groupTreeDirty = 2;
}
m_blockSize = 10240; // ###tw This really ought to be the most
// efficient file reading size for the current
// operating system.
m_block = NULL;
while (!m_block && (m_blockSize >= 512))
{
m_block = new char[m_blockSize + 1];
if (!m_block)
m_blockSize /= 2;
}
if (!m_block)
return MK_OUT_OF_MEMORY;
m_groupTree = msg_GroupRecord::Create(NULL, NULL, 0, 0, 0);
if (!m_groupTree)
return MK_OUT_OF_MEMORY;
return ReadInitialPart();
}
int
MSG_NewsHost::ReadInitialPart()
{
OpenGroupFile();
if (!m_groupFile) return -1;
int32 length = XP_FileRead(m_block, m_blockSize, m_groupFile);
if (length < 0) length = 0;
m_block[length] = '\0';
int32 version = 0;
char* ptr;
char* endptr = NULL;
for (ptr = m_block ; *ptr ; ptr = endptr + 1) {
while (*ptr == CR || *ptr == LF) ptr++;
endptr = XP_STRCHR(ptr, LINEBREAK_START);
if (!endptr) break;
*endptr = '\0';
if (ptr[0] == '#' || ptr[0] == '\0') {
continue; // Skip blank lines and comments.
}
if (XP_STRCMP(ptr, "begingroups") == 0) {
m_fileStart = endptr - m_block;
break;
}
char* ptr2 = XP_STRCHR(ptr, '=');
if (!ptr2) continue;
*ptr2++ = '\0';
if (XP_STRCMP(ptr, "lastgroupdate") == 0) {
m_lastgroupdate = strtol(ptr2, NULL, 16);
} else if (XP_STRCMP(ptr, "firstnewdate") == 0) {
m_firstnewdate = strtol(ptr2, NULL, 16);
} else if (XP_STRCMP(ptr, "uniqueid") == 0) {
m_uniqueId = strtol(ptr2, NULL, 16);
} else if (XP_STRCMP(ptr, "pushauth") == 0) {
m_pushAuth = strtol(ptr2, NULL, 16);
} else if (XP_STRCMP(ptr, "version") == 0) {
version = strtol(ptr2, NULL, 16);
}
}
m_block[0] = '\0';
if (version != 1) {
// The file got clobbered or damaged somehow. Throw it away.
#ifdef DEBUG_bienvenu
if (length > 0)
XP_ASSERT(FALSE); // this really shouldn't happen, right?
#endif
OpenGroupFile(XP_FILE_WRITE_BIN);
XP_ASSERT(m_groupFile);
if (!m_groupFile) return -1;
OpenGroupFile();
XP_ASSERT(m_groupFile);
if (!m_groupFile) return -1;
m_groupTreeDirty = 2;
m_fileStart = 0;
m_fileSize = 0;
}
m_groupTree->SetFileOffset(m_fileStart);
return 0;
}
int
MSG_NewsHost::CreateFileHeader()
{
PR_snprintf(m_block, m_blockSize,
"# Netscape newshost information file." LINEBREAK
"# This is a generated file! Do not edit." LINEBREAK
"" LINEBREAK
"version=1" LINEBREAK
"newsrcname=%s" LINEBREAK
"lastgroupdate=%08lx" LINEBREAK
"firstnewdate=%08lx" LINEBREAK
"uniqueid=%08lx" LINEBREAK
"pushauth=%1x" LINEBREAK
"" LINEBREAK
"begingroups",
m_filename, (long) m_lastgroupdate, (long) m_firstnewdate, (long) m_uniqueId,
m_pushAuth);
return XP_STRLEN(m_block);
}
int
MSG_NewsHost::SaveHostInfo()
{
int status = 0;
msg_GroupRecord* grec;
XP_File in = NULL;
XP_File out = NULL;
char* blockcomma = NULL;
char* ptrcomma = NULL;
char* filename = NULL;
int length = CreateFileHeader();
XP_ASSERT(length < m_blockSize - 50);
if (m_inhaled || length != m_fileStart) {
m_groupTreeDirty = 2;
}
if (m_groupTreeDirty < 0) {
// Uh, somehow we set the sign bit, probably some routine returned an
// error status. This will probably never happen. But if it does,
// we should just make sure to write things out in the most paranoid
// fashion.
m_groupTreeDirty = 2;
}
if (m_groupTreeDirty < 2) {
// Only bits and pieces are dirty. Write them out without writing the
// whole file.
OpenGroupFile();
if (!m_groupFile) {
status = MK_UNABLE_TO_OPEN_NEWSRC;
goto FAIL;
}
XP_FileSeek(m_groupFile, 0, SEEK_SET);
XP_FileWrite(m_block, length, m_groupFile);
char* ptr = NULL;
for (grec = m_groupTree->GetChildren();
grec;
grec = grec->GetNextAlphabetic()) {
if (grec->IsDirty()) {
if (grec->GetFileOffset() < m_fileStart) {
m_groupTreeDirty = 2;
break;
}
ptr = grec->GetSaveString();
if (!ptr) {
status = MK_OUT_OF_MEMORY;
goto FAIL;
}
length = XP_STRLEN(ptr);
if (length >= m_blockSize) {
char* tmp = new char [length + 512];
if (!tmp) {
status = MK_OUT_OF_MEMORY;
goto FAIL;
}
delete [] m_block;
m_block = tmp;
m_blockSize = length + 512;
}
// Read the old data, and make sure the old line was the
// same length as the new one. If not, set the dirty flag
// to 2, so we end up writing the whole file, below.
XP_FileSeek(m_groupFile, grec->GetFileOffset(), SEEK_SET);
int l = XP_FileRead(m_block, length, m_groupFile);
if (l != length || m_block[length - 1] != ptr[length - 1]) {
m_groupTreeDirty = 2;
break;
}
m_block[l] = '\0';
char* p1 = XP_STRCHR(ptr, LINEBREAK_START);
char* p2 = XP_STRCHR(m_block, LINEBREAK_START);
XP_ASSERT(p1);
if (!p1 || !p2 || (p1 - ptr) != (p2 - m_block)) {
m_groupTreeDirty = 2;
break;
}
XP_ASSERT(grec->GetFileOffset() > 100); // header is at least 100 bytes long
XP_FileSeek(m_groupFile, grec->GetFileOffset(), SEEK_SET);
XP_FileWrite(ptr, XP_STRLEN(ptr), m_groupFile);
XP_FREE(ptr);
ptr = NULL;
}
}
FREEIF(ptr);
}
if (m_groupTreeDirty >= 2) {
// We need to rewrite the whole file.
filename = PR_smprintf("%s/tempinfo", GetDBDirName());
if (!filename) return MK_OUT_OF_MEMORY;
int32 position = 0;
if (m_groupFile)
{
XP_FileClose(m_groupFile);
m_groupFile = NULL;
}
out = XP_FileOpen(filename, xpXoverCache, XP_FILE_WRITE_BIN);
if (!out) return MK_MIME_ERROR_WRITING_FILE;
length = CreateFileHeader(); // someone probably has hosed m_block - so regen
status = XP_FileWrite(m_block, length, out);
if (status < 0) goto FAIL;
position += length;
m_fileStart = position;
m_groupTree->SetFileOffset(m_fileStart);
status = XP_FileWrite(LINEBREAK, LINEBREAK_LEN, out);
if (status < 0) goto FAIL;
position += LINEBREAK_LEN;
blockcomma = NULL;
if (!m_inhaled) {
in = XP_FileOpen(m_hostinfofilename, xpXoverCache, XP_FILE_READ);
if (!in) {
status = MK_MIME_ERROR_WRITING_FILE;
goto FAIL;
}
do {
m_block[0] = '\0';
XP_FileReadLine(m_block, m_blockSize, in);
} while (m_block[0] &&
XP_STRNCMP(m_block, "begingroups", 11) != 0);
m_block[0] = '\0';
XP_FileReadLine(m_block, m_blockSize, in);
blockcomma = XP_STRCHR(m_block, ',');
if (blockcomma) *blockcomma = '\0';
}
for (grec = m_groupTree->GetChildren();
grec;
grec = grec->GetNextAlphabetic()) {
char* ptr = grec->GetSaveString();
if (!ptr) {
status = MK_OUT_OF_MEMORY;
goto FAIL;
}
ptrcomma = XP_STRCHR(ptr, ',');
XP_ASSERT(ptrcomma);
if (!ptrcomma) {
status = -1;
goto FAIL;
}
*ptrcomma = '\0';
while (blockcomma &&
msg_GroupRecord::GroupNameCompare(m_block, ptr) < 0) {
*blockcomma = ',';
XP_StripLine(m_block);
length = XP_STRLEN(m_block);
status = XP_FileWrite(m_block, length, out);
if (status < 0) goto FAIL;
position += length;
status = XP_FileWrite(LINEBREAK, LINEBREAK_LEN, out);
if (status < 0) goto FAIL;
position += LINEBREAK_LEN;
m_block[0] = '\0';
XP_FileReadLine(m_block, m_blockSize, in);
blockcomma = XP_STRCHR(m_block, ',');
if (blockcomma) *blockcomma = '\0';
}
if (blockcomma && XP_STRCMP(m_block, ptr) == 0) {
m_block[0] = '\0';
XP_FileReadLine(m_block, m_blockSize, in);
blockcomma = XP_STRCHR(m_block, ',');
if (blockcomma) *blockcomma = '\0';
}
grec->SetFileOffset(position);
*ptrcomma = ',';
length = XP_STRLEN(ptr);
// if the group doesn't exist on the server and has no children,
// don't write it out.
if (!grec->DoesNotExistOnServer() || grec->GetNumKids() > 0)
{
status = XP_FileWrite(ptr, length, out);
position += length;
}
else
status = 0;
XP_FREE(ptr);
if (status < 0) goto FAIL;
}
if (blockcomma) {
*blockcomma = ',';
do {
XP_StripLine(m_block);
length = XP_STRLEN(m_block);
status = XP_FileWrite(m_block, length, out);
if (status < 0) goto FAIL;
position += length;
status = XP_FileWrite(LINEBREAK, LINEBREAK_LEN, out);
if (status < 0) goto FAIL;
position += LINEBREAK_LEN;
m_block[0] = '\0';
} while (XP_FileReadLine(m_block, m_blockSize, in) && *m_block);
}
if (in) {
XP_FileClose(in);
in = NULL;
}
XP_FileClose(out);
out = NULL;
XP_FileRename(filename, xpXoverCache,
m_hostinfofilename, xpXoverCache);
m_fileSize = position;
}
m_groupTreeDirty = 0;
FAIL:
if (in) XP_FileClose(in);
if (out) XP_FileClose(out);
if (filename) XP_FREE(filename);
m_block[0] = '\0';
return status;
}
struct InhaleState {
msg_GroupRecord* tree;
int32 position;
msg_GroupRecord* onlyIfChild;
msg_GroupRecord* lastInhaled;
char lastfullname[512];
};
int32
MSG_NewsHost::InhaleLine(char* line, uint32 length, void* closure)
{
int status = 0;
InhaleState* state = (InhaleState*) closure;
int32 position = state->position;
state->position += length;
char* lastdot;
char* endptr = line + length;
char* comma;
for (comma = line ; comma < endptr ; comma++) {
if (*comma == ',') break;
}
if (comma >= endptr) return 0;
*comma = '\0';
lastdot = XP_STRRCHR(line, '.');
msg_GroupRecord* parent;
msg_GroupRecord* child;
if (lastdot) {
*lastdot = '\0';
// Try to find the parent of this line. Very often, it will be
// the last line we read, or some ancestor of that line.
parent = NULL;
if (state->lastInhaled && XP_STRNCMP(line, state->lastfullname,
lastdot - line) == 0) {
char c = state->lastfullname[lastdot - line];
if (c == '\0') parent = state->lastInhaled;
else if (c == '.') {
parent = state->lastInhaled;
char* ptr = state->lastfullname + (lastdot - line);
XP_ASSERT(parent);
while (parent && ptr) {
parent = parent->GetParent();
XP_ASSERT(parent);
ptr = XP_STRCHR(ptr + 1, '.');
}
}
}
if (!parent) parent = state->tree->FindDescendant(line);
*lastdot = '.';
XP_ASSERT(parent);
if (!parent) {
status = -1;
goto DONE;
}
} else {
parent = state->tree;
lastdot = line - 1;
}
child = parent->FindDescendant(lastdot + 1);
*comma = ',';
if (child) {
// It's already in memory.
child->SetFileOffset(position);
} else {
child = msg_GroupRecord::Create(parent, line, endptr - line, position);
if (!child) {
status = MK_OUT_OF_MEMORY;
goto DONE;
}
}
if (state->onlyIfChild) {
msg_GroupRecord* tmp;
for (tmp = child; tmp ; tmp = tmp->GetParent()) {
if (tmp == state->onlyIfChild) break;
}
if (tmp == NULL) status = -2; // Indicates we're done.
}
XP_ASSERT(comma - line < sizeof(state->lastfullname));
if (comma - line < sizeof(state->lastfullname)) {
XP_STRNCPY_SAFE(state->lastfullname, line, comma - line + 1);
state->lastfullname[comma - line] = '\0';
state->lastInhaled = child;
}
DONE:
if (comma) *comma = ',';
return status;
}
int
MSG_NewsHost::Inhale(XP_Bool force)
{
if (m_groupTreeDirty) SaveHostInfo();
if (force) {
while (m_groupTree->GetChildren()) {
delete m_groupTree->GetChildren();
}
m_inhaled = FALSE;
}
XP_ASSERT(!m_inhaled);
if (m_inhaled) return -1;
int status = 0;
OpenGroupFile();
if (!m_groupFile) return -1;
XP_FileSeek(m_groupFile, 0, SEEK_SET);
status = ReadInitialPart();
if (status < 0) return status;
XP_FileSeek(m_groupFile, m_fileStart, SEEK_SET);
InhaleState state;
state.tree = m_groupTree;
state.position = m_fileStart;
state.onlyIfChild = NULL;
state.lastInhaled = NULL;
char* buffer = NULL;
uint32 buffer_size = 0;
uint32 buffer_fp = 0;
do {
status = XP_FileRead(m_block, m_blockSize, m_groupFile);
if (status <= 0) break;
status = msg_LineBuffer(m_block, status,
&buffer, &buffer_size, &buffer_fp,
FALSE,
#ifdef XP_OS2
(int32 (_Optlink*) (char*,uint32,void*))
#endif
MSG_NewsHost::InhaleLine, &state);
} while (status >= 0);
if (status >= 0 && buffer_fp > 0) {
status = InhaleLine(buffer, buffer_fp, &state);
}
FREEIF(buffer);
m_block[0] = '\0';
if (status >= 0) m_inhaled = TRUE;
return status;
}
int
MSG_NewsHost::Exhale()
{
XP_ASSERT(m_inhaled);
if (!m_inhaled) return -1;
int status = SaveHostInfo();
while (m_groupTree->GetChildren()) {
delete m_groupTree->GetChildren();
}
m_inhaled = FALSE;
return status;
}
int
MSG_NewsHost::EmptyInhale()
{
XP_ASSERT(!m_inhaled);
if (m_inhaled) return -1;
while (m_groupTree->GetChildren()) {
delete m_groupTree->GetChildren();
}
m_inhaled = TRUE;
return 0;
}
MSG_FolderInfoNews*
MSG_NewsHost::FindGroup(const char* name)
{
if (m_groups == NULL) return NULL;
int n = m_groups->GetSize();
for (int i=0 ; i<n ; i++) {
MSG_FolderInfoNews* info = (MSG_FolderInfoNews*) (*m_groups)[i];
if (XP_STRCMP(info->GetNewsgroupName(), name) == 0) {
return info;
}
}
return NULL;
}
MSG_FolderInfoNews *MSG_NewsHost::AddGroup(const char *groupName)
{
MSG_FolderInfoNews *newsInfo = NULL;
MSG_FolderInfoCategoryContainer *categoryContainer = NULL;
char* containerName = NULL;
XP_Bool needpaneupdate = FALSE;
msg_GroupRecord* group = FindOrCreateGroup(groupName);
if (!group) goto DONE; // Out of memory.
if (!group->IsCategoryContainer() && group->IsCategory()) {
msg_GroupRecord *container = group->GetCategoryContainer();
XP_ASSERT(container);
if (!container) goto DONE;
containerName = container->GetFullName();
if (!containerName) goto DONE; // Out of memory.
newsInfo = FindGroup(containerName);
categoryContainer = (MSG_FolderInfoCategoryContainer *) newsInfo;
// if we're not subscribed to container, do that instead.
if (!newsInfo || !newsInfo->IsSubscribed())
{
groupName = containerName;
group = FindOrCreateGroup(groupName);
}
}
// no need to diddle folder pane for new categories.
needpaneupdate = m_hostinfo && !group->IsCategory();
newsInfo = FindGroup(groupName);
if (newsInfo) { // seems to be already added
if (!newsInfo->IsSubscribed()) {
newsInfo->Subscribe(TRUE);
XPPtrArray* infolist = (XPPtrArray*) m_hostinfo->GetSubFolders();
// don't add it if it's already in the list!
if (infolist->FindIndex(0, newsInfo) == -1)
infolist->Add(newsInfo);
} else {
goto DONE;
}
} else {
char* groupLine = PR_smprintf("%s:", groupName);
if (!groupLine) {
goto DONE; // Out of memory.
}
// this will add and auto-subscribe - OK, a cheap hack.
if (ProcessLine(groupLine, XP_STRLEN(groupLine)) == 0) {
// groups are added at end so look there first...
newsInfo = (MSG_FolderInfoNews *)
m_groups->GetAt(m_groups->GetSize() - 1);
if (XP_STRCMP(newsInfo->GetNewsgroupName(), groupName)) {
newsInfo = FindGroup(groupName);
XP_ASSERT(newsInfo);
}
}
XP_FREE(groupLine);
}
XP_ASSERT(newsInfo);
if (!newsInfo) goto DONE;
if (group->IsCategoryContainer()) {
// Go add all of our categories to the newsrc.
AssureAllDescendentsLoaded(group);
msg_GroupRecord* end = group->GetSiblingOrAncestorSibling();
msg_GroupRecord* child;
for (child = group->GetNextAlphabetic() ;
child != end ;
child = child->GetNextAlphabetic()) {
XP_ASSERT(child);
if (!child) break;
char* fullname = child->GetFullName();
if (!fullname) break;
MSG_FolderInfoNews* info = FindGroup(fullname);
if (info) {
if (!info->IsSubscribed()) {
info->Subscribe(TRUE);
}
} else {
char* groupLine = PR_smprintf("%s:", fullname);
if (groupLine) {
ProcessLine(groupLine, XP_STRLEN(groupLine));
XP_FREE(groupLine);
}
}
delete [] fullname;
}
if (newsInfo->GetType() != FOLDER_CATEGORYCONTAINER)
{
newsInfo = SwitchNewsToCategoryContainer(newsInfo);
}
XP_ASSERT(newsInfo->GetType() == FOLDER_CATEGORYCONTAINER);
if (newsInfo->GetType() == FOLDER_CATEGORYCONTAINER) {
newsInfo->SetFlag(MSG_FOLDER_FLAG_ELIDED);
MSG_FolderInfoCategoryContainer *catContainer =
(MSG_FolderInfoCategoryContainer *) newsInfo;
catContainer->BuildCategoryTree(catContainer, groupName, group,
2, m_master);
}
}
else if (group->IsCategory() && categoryContainer)
{
categoryContainer->AddToCategoryTree(categoryContainer, groupName, group, m_master);
}
if (needpaneupdate && newsInfo)
m_master->BroadcastFolderAdded (newsInfo);
MarkDirty();
DONE:
if (containerName) delete [] containerName;
return newsInfo;
}
MSG_FolderInfoNews *MSG_NewsHost::SwitchNewsToCategoryContainer(MSG_FolderInfoNews *newsInfo)
{
int groupIndex = m_groups->FindIndex(0, newsInfo);
if (groupIndex != -1)
{
MSG_FolderInfoCategoryContainer *newCatCont = newsInfo->CloneIntoCategoryContainer();
// slip the category container where the newsInfo was.
m_groups->SetAt(groupIndex, newCatCont);
XPPtrArray* infoList = (XPPtrArray*) m_hostinfo->GetSubFolders();
// replace in folder pane server list as well.
groupIndex = infoList->FindIndex(0, newsInfo);
if (groupIndex != -1)
infoList->SetAt(groupIndex, newCatCont);
newsInfo = newCatCont;
}
return newsInfo;
}
MSG_FolderInfoNews *MSG_NewsHost::SwitchCategoryContainerToNews(MSG_FolderInfoCategoryContainer *catContainerInfo)
{
MSG_FolderInfoNews *retInfo = NULL;
int groupIndex = m_groups->FindIndex(0, catContainerInfo);
if (groupIndex != -1)
{
MSG_FolderInfoNews *rootCategory = catContainerInfo->GetRootCategory();
// slip the root category container where the category container was.
m_groups->SetAt(groupIndex, rootCategory);
XPPtrArray* infoList = (XPPtrArray*) m_hostinfo->GetSubFolders();
// replace in folder pane server list as well.
groupIndex = infoList->FindIndex(0, catContainerInfo);
if (groupIndex != -1)
infoList->SetAt(groupIndex, rootCategory);
retInfo = rootCategory;
// this effectively leaks the category container, but I don't think that's a problem
}
return retInfo;
}
void MSG_NewsHost::RemoveGroup (MSG_FolderInfoNews *newsInfo)
{
if (newsInfo && newsInfo->IsSubscribed())
{
newsInfo->Subscribe(FALSE);
m_master->BroadcastFolderDeleted (newsInfo);
XPPtrArray* infolist = (XPPtrArray*) m_hostinfo->GetSubFolders();
infolist->Remove(newsInfo);
}
}
void
MSG_NewsHost::RemoveGroup(const char* groupName)
{
MSG_FolderInfoNews *newsInfo = NULL;
newsInfo = FindGroup(groupName);
RemoveGroup (newsInfo);
}
#ifdef DEBUG_terry
// #define XP_WIN16
#endif
const char*
MSG_NewsHost::GetDBDirName()
{
if (!m_filename) return NULL;
if (!m_dbfilename) {
#if defined(XP_WIN16) || defined(XP_OS2)
const int MAX_HOST_NAME_LEN = 8;
#elif defined(XP_MAC)
const int MAX_HOST_NAME_LEN = 25;
#else
const int MAX_HOST_NAME_LEN = 55;
#endif
// Given a hostname, use either that hostname, if it fits on our
// filesystem, or a hashified version of it, if the hostname is too
// long to fit.
char hashedname[MAX_HOST_NAME_LEN + 1];
XP_Bool needshash = XP_STRLEN(m_filename) > MAX_HOST_NAME_LEN;
#if defined(XP_WIN16) || defined(XP_OS2)
if (!needshash) {
needshash = XP_STRCHR(m_filename, '.') != NULL ||
XP_STRCHR(m_filename, ':') != NULL;
}
#endif
XP_STRNCPY_SAFE(hashedname, m_filename, MAX_HOST_NAME_LEN + 1);
if (needshash) {
PR_snprintf(hashedname + MAX_HOST_NAME_LEN - 8, 9, "%08lx",
(unsigned long) XP_StringHash2(m_filename));
}
m_dbfilename = new char [XP_STRLEN(hashedname) + 15];
XP_STRCPY(m_dbfilename, HG38376 "host-");
XP_STRCAT(m_dbfilename, hashedname);
char* ptr = XP_STRCHR(m_dbfilename, ':');
if (ptr) *ptr = '.'; // WinFE doesn't like colons in filenames.
XP_MakeDirectory(m_dbfilename, xpXoverCache);
}
return m_dbfilename;
}
#ifdef DEBUG_terry
// #undef XP_WIN16
#endif
int32
MSG_NewsHost::GetNumGroupsNeedingCounts()
{
if (!m_groups) return 0;
int num = m_groups->GetSize();
int32 result = 0;
for (int i=0 ; i<num ; i++) {
MSG_FolderInfoNews* info = (MSG_FolderInfoNews*) ((*m_groups)[i]);
if (info->GetWantNewTotals() && info->IsSubscribed()) {
result++;
}
}
return result;
}
char*
MSG_NewsHost::GetFirstGroupNeedingCounts()
{
if (!m_groups) return NULL;
int num = m_groups->GetSize();
for (int i=0 ; i<num ; i++) {
MSG_FolderInfoNews* info = (MSG_FolderInfoNews*) ((*m_groups)[i]);
if (info->GetWantNewTotals() && info->IsSubscribed()) {
info->SetWantNewTotals(FALSE);
return XP_STRDUP(info->GetNewsgroupName());
}
}
return NULL;
}
char*
MSG_NewsHost::GetFirstGroupNeedingExtraInfo()
{
msg_GroupRecord* grec;
for (grec = m_groupTree->GetChildren(); grec; grec = grec->GetNextAlphabetic()) {
if (grec && grec->NeedsExtraInfo()) {
char *fullName = grec->GetFullName();
char *ret = XP_STRDUP(fullName);
delete [] fullName;
return ret;
}
}
return NULL;
}
void
MSG_NewsHost::SetWantNewTotals(XP_Bool value)
{
if (!m_groups) return;
int n = m_groups->GetSize();
for (int i=0 ; i<n ; i++) {
MSG_FolderInfoNews* info = (MSG_FolderInfoNews*) ((*m_groups)[i]);
info->SetWantNewTotals(value);
}
}
XP_Bool MSG_NewsHost::NeedsExtension (const char * /*extension*/)
{
//###phil need to flesh this out
XP_Bool needed = m_supportsExtensions;
return needed;
}
void MSG_NewsHost::AddExtension (const char *ext)
{
if (!QueryExtension(ext))
{
char *ourExt = XP_STRDUP (ext);
if (ourExt)
m_supportedExtensions.Add(ourExt);
}
}
XP_Bool MSG_NewsHost::QueryExtension (const char *ext)
{
for (int i = 0; i < m_supportedExtensions.GetSize(); i++)
if (!XP_STRCMP(ext, (char*) m_supportedExtensions.GetAt(i)))
return TRUE;
return FALSE;
}
void MSG_NewsHost::AddSearchableGroup (const char *group)
{
if (!QuerySearchableGroup(group))
{
char *ourGroup = XP_STRDUP (group);
if (ourGroup)
{
// strip off character set spec
char *space = XP_STRCHR(ourGroup, ' ');
if (space)
*space = '\0';
m_searchableGroups.Add(ourGroup);
space++; // walk over to the start of the charsets
// Add the group -> charset association.
XP_Puthash(m_searchableGroupCharsets, ourGroup, space);
}
}
}
XP_Bool MSG_NewsHost::QuerySearchableGroup (const char *group)
{
for (int i = 0; i < m_searchableGroups.GetSize(); i++)
{
const char *searchableGroup = (const char*) m_searchableGroups.GetAt(i);
char *starInSearchableGroup = NULL;
if (!XP_STRCMP(searchableGroup, "*"))
return TRUE; // everything is searchable
else if (NULL != (starInSearchableGroup = XP_STRCHR(searchableGroup, '*')))
{
if (!XP_STRNCASECMP(group, searchableGroup, XP_STRLEN(searchableGroup)-2))
return TRUE; // this group is in a searchable hierarchy
}
else if (!XP_STRCASECMP(group, searchableGroup))
return TRUE; // this group is individually searchable
}
return FALSE;
}
// ### mwelch This should have been merged into one routine with QuerySearchableGroup,
// but with two interfaces.
const char *
MSG_NewsHost::QuerySearchableGroupCharsets(const char *group)
{
// Very similar to the above, but this time we look up charsets.
const char *result = NULL;
XP_Bool gotGroup = FALSE;
const char *searchableGroup = NULL;
for (int i = 0; (i < m_searchableGroups.GetSize()) && (!gotGroup); i++)
{
searchableGroup = (const char*) m_searchableGroups.GetAt(i);
char *starInSearchableGroup = NULL;
if (!XP_STRCMP(searchableGroup, "*"))
gotGroup = TRUE; // everything is searchable
else if (NULL != (starInSearchableGroup = XP_STRCHR(searchableGroup, '*')))
{
if (!XP_STRNCASECMP(group, searchableGroup, XP_STRLEN(searchableGroup)-2))
gotGroup = TRUE; // this group is in a searchable hierarchy
}
else if (!XP_STRCASECMP(group, searchableGroup))
gotGroup = TRUE; // this group is individually searchable
}
if (gotGroup)
{
// Look up the searchable group for its supported charsets
result = (const char *) XP_Gethash(m_searchableGroupCharsets, searchableGroup, NULL);
}
return result;
}
void MSG_NewsHost::AddSearchableHeader (const char *header)
{
if (!QuerySearchableHeader(header))
{
char *ourHeader = XP_STRDUP(header);
if (ourHeader)
m_searchableHeaders.Add(ourHeader);
}
}
XP_Bool MSG_NewsHost::QuerySearchableHeader(const char *header)
{
for (int i = 0; i < m_searchableHeaders.GetSize(); i++)
if (!XP_STRNCASECMP(header, (char*) m_searchableHeaders.GetAt(i), XP_STRLEN(header)))
return TRUE;
return FALSE;
}
void MSG_NewsHost::AddPropertyForGet (const char *property, const char *value)
{
char *tmp = NULL;
tmp = XP_STRDUP(property);
if (tmp)
m_propertiesForGet.Add (tmp);
tmp = XP_STRDUP(value);
if (tmp)
m_valuesForGet.Add (tmp);
}
const char *MSG_NewsHost::QueryPropertyForGet (const char *property)
{
for (int i = 0; i < m_propertiesForGet.GetSize(); i++)
if (!XP_STRCASECMP(property, (const char *) m_propertiesForGet.GetAt(i)))
return (const char *) m_valuesForGet.GetAt(i);
return NULL;
}
void MSG_NewsHost::SetPushAuth(XP_Bool value)
{
if (m_pushAuth != value)
{
m_pushAuth = value;
m_groupTreeDirty |= 1;
}
}
const char*
MSG_NewsHost::GetURLBase()
{
if (!m_urlbase) {
m_urlbase = PR_smprintf("%s://%s", HG38262 "news",
getNameAndPort());
}
return m_urlbase;
}
int MSG_NewsHost::RemoveHost()
{
if (m_groupFile) {
XP_FileClose(m_groupFile);
m_groupFile = NULL;
}
m_master->GetFolderTree()->RemoveSubFolder(m_hostinfo);
m_master->GetHostTable()->RemoveEntry(this);
m_dirty = 0;
if (m_writetimer) {
FE_ClearTimeout(m_writetimer);
m_writetimer = NULL;
}
XP_FileRemove(GetNewsrcFileName(), HG37252 xpNewsRC);
// Now go do our best to remove the entire newshost directory and all of
// its contents. I would love to use XP_RemoveDirectoryRecursive(), but
// selmer seems to have only implemented it on WinFE. Ahem.
int lastcount = -1;
int count = -1;
do {
XP_Dir dir = XP_OpenDir(GetDBDirName(), xpXoverCache);
if (!dir) break;
lastcount = count;
count = 0;
XP_DirEntryStruct *entry = NULL;
for (entry = XP_ReadDir(dir); entry; entry = XP_ReadDir(dir)) {
count++;
const char* name = entry->d_name;
if (name[0] == '.') continue;
char* tmp = PR_smprintf("%s/%s", GetDBDirName(), name);
if (tmp) {
XP_FileRemove(tmp, xpXoverCache);
XP_FREE(tmp);
}
}
XP_CloseDir(dir);
} while (lastcount != count); // Keep doing it over and over, until
// we get the same number of entries in
// the dir. This is because I do not
// trust XP_ReadDir() to do the right
// thing if I delete files out from
// under it.
XP_RemoveDirectory(GetDBDirName(), xpXoverCache);
// Tell netlib to close any connections we might have to
// this news host.
NET_OnNewsHostDeleted (m_hostname);
m_master->BroadcastFolderDeleted (m_hostinfo);
delete this;
return 0;
}
char*
MSG_NewsHost::GetPrettyName(const char* groupname)
{
msg_GroupRecord* group = FindOrCreateGroup(groupname);
if (group) {
const char* orig = group->GetPrettyName();
if (orig) {
char* result = new char [XP_STRLEN(orig) + 1];
if (result) {
XP_STRCPY(result, orig);
return result;
}
}
}
return NULL;
}
int
MSG_NewsHost::SetPrettyName(const char* groupname, const char* prettyname)
{
msg_GroupRecord* group = FindOrCreateGroup(groupname);
if (!group) return MK_OUT_OF_MEMORY;
int status = group->SetPrettyName(prettyname);
if (status > 0)
{
MSG_FolderInfoNews *newsFolder = FindGroup(groupname);
// make news folder forget prettyname so it will query again
if (newsFolder)
newsFolder->ClearPrettyName();
}
m_groupTreeDirty |= status;
return status;
}
time_t
MSG_NewsHost::GetAddTime(const char* groupname)
{
msg_GroupRecord* group = FindOrCreateGroup(groupname);
if (!group) return 0;
return group->GetAddTime();
}
int32
MSG_NewsHost::GetUniqueID(const char* groupname)
{
msg_GroupRecord* group = FindOrCreateGroup(groupname);
if (!group) return 0;
return group->GetUniqueID();
}
XP_Bool
MSG_NewsHost::IsCategory(const char* groupname)
{
msg_GroupRecord* group = FindOrCreateGroup(groupname);
if (!group) return FALSE;
return group->IsCategory();
}
XP_Bool
MSG_NewsHost::IsCategoryContainer(const char* groupname)
{
msg_GroupRecord* group = FindOrCreateGroup(groupname);
if (!group) return FALSE;
return group->IsCategoryContainer();
}
int
MSG_NewsHost::SetIsCategoryContainer(const char* groupname, XP_Bool value)
{
msg_GroupRecord* group = FindOrCreateGroup(groupname);
if (!group) return MK_OUT_OF_MEMORY;
int status = group->SetIsCategoryContainer(value);
m_groupTreeDirty |= status;
if (status > 0)
{
MSG_FolderInfoNews *newsFolder = FindGroup(groupname);
// make news folder have correct category container flag
if (newsFolder)
{
if (value)
{
newsFolder->SetFlag(MSG_FOLDER_FLAG_CAT_CONTAINER);
SwitchNewsToCategoryContainer(newsFolder);
}
else
{
if (newsFolder->GetType() == FOLDER_CATEGORYCONTAINER)
{
MSG_FolderInfoCategoryContainer *catCont = (MSG_FolderInfoCategoryContainer *) newsFolder;
newsFolder = SwitchCategoryContainerToNews(catCont);
}
if (newsFolder)
{
newsFolder->ClearFlag(MSG_FOLDER_FLAG_CAT_CONTAINER);
newsFolder->ClearFlag(MSG_FOLDER_FLAG_CATEGORY);
}
}
}
}
return status;
}
int
MSG_NewsHost::SetGroupNeedsExtraInfo(const char *groupname, XP_Bool value)
{
msg_GroupRecord* group = FindOrCreateGroup(groupname);
if (!group) return MK_OUT_OF_MEMORY;
int status = group->SetNeedsExtraInfo(value);
m_groupTreeDirty |= status;
return status;
}
char*
MSG_NewsHost::GetCategoryContainer(const char* groupname)
{
msg_GroupRecord* group = FindOrCreateGroup(groupname);
if (group) {
group = group->GetCategoryContainer();
if (group) return group->GetFullName();
}
return NULL;
}
MSG_FolderInfoNews *
MSG_NewsHost::GetCategoryContainerFolderInfo(const char *groupname)
{
MSG_FolderInfoNews *ret = NULL;
// because GetCategoryContainer returns NULL for a category container...
msg_GroupRecord *group = FindOrCreateGroup(groupname);
if (group->IsCategoryContainer())
return FindGroup(groupname);
char *categoryContainerName = GetCategoryContainer(groupname);
if (categoryContainerName)
{
ret = FindGroup(categoryContainerName);
delete [] categoryContainerName;
}
return ret;
}
XP_Bool
MSG_NewsHost::IsProfile(const char* groupname)
{
msg_GroupRecord* group = FindOrCreateGroup(groupname);
if (!group) return FALSE;
return group->IsProfile();
}
int
MSG_NewsHost::SetIsProfile(const char* groupname, XP_Bool value)
{
msg_GroupRecord* group = FindOrCreateGroup(groupname);
if (!group) return MK_OUT_OF_MEMORY;
int status = group->SetIsProfile(value);
m_groupTreeDirty |= status;
return status;
}
XP_Bool
MSG_NewsHost::IsGroup(const char* groupname)
{
msg_GroupRecord* group = FindOrCreateGroup(groupname);
if (!group) return FALSE;
return group->IsGroup();
}
int
MSG_NewsHost::SetIsGroup(const char* groupname, XP_Bool value)
{
msg_GroupRecord* group = FindOrCreateGroup(groupname);
if (!group) return MK_OUT_OF_MEMORY;
int status = group->SetIsGroup(value);
m_groupTreeDirty |= status;
return status;
}
XP_Bool
MSG_NewsHost::IsHTMLOk(const char* groupname)
{
msg_GroupRecord* group = FindOrCreateGroup(groupname);
if (!group) return FALSE;
if (group->IsHTMLOKGroup()) return TRUE;
for ( ; group ; group = group->GetParent()) {
if (group->IsHTMLOKTree()) return TRUE;
}
return FALSE;
}
XP_Bool
MSG_NewsHost::IsHTMLOKGroup(const char* groupname)
{
msg_GroupRecord* group = FindOrCreateGroup(groupname);
if (!group) return FALSE;
return group->IsHTMLOKGroup();
}
int
MSG_NewsHost::SetIsHTMLOKGroup(const char* groupname, XP_Bool value)
{
msg_GroupRecord* group = FindOrCreateGroup(groupname);
if (!group) return MK_OUT_OF_MEMORY;
int status = group->SetIsHTMLOKGroup(value);
m_groupTreeDirty |= status;
return status;
}
XP_Bool
MSG_NewsHost::IsHTMLOKTree(const char* groupname)
{
msg_GroupRecord* group = FindOrCreateGroup(groupname);
if (!group) return FALSE;
return group->IsHTMLOKTree();
}
int
MSG_NewsHost::SetIsHTMLOKTree(const char* groupname, XP_Bool value)
{
msg_GroupRecord* group = FindOrCreateGroup(groupname);
if (!group) return MK_OUT_OF_MEMORY;
int status = group->SetIsHTMLOKTree(value);
m_groupTreeDirty |= status;
return status;
}
msg_GroupRecord*
MSG_NewsHost::FindGroupInBlock(msg_GroupRecord* parent,
const char* groupname,
int32* comp)
{
char* ptr;
char* ptr2;
char* tmp;
RESTART:
ptr = m_block;
*comp = 0;
for (;;) {
ptr = XP_STRCHR(ptr, LINEBREAK_START);
ptr2 = ptr ? XP_STRCHR(ptr, ',') : 0;
if (!ptr2) {
if (*comp != 0) return NULL;
// Yikes. The whole string, and we can't even find a legitimate
// beginning of a real line. Grow the buffer until we can.
if (m_fileSize - m_fileStart <= m_blockSize) {
// Oh, well, maybe the file is just empty.
*comp = 0;
return NULL;
}
if (int32(XP_STRLEN(m_block)) < m_blockSize) goto RELOAD;
goto GROWBUFFER;
}
while (*ptr == CR || *ptr == LF) ptr++;
*ptr2 = '\0';
int32 c = msg_GroupRecord::GroupNameCompare(groupname, ptr);
*ptr2 = ',';
if (c < 0) {
if (*comp > 0) {
// This group isn't in the file, since earlier compares said
// "look later" and this one says "look earlier".
*comp = 0;
return NULL;
}
*comp = c;
return NULL; // This group is somewhere before this chunk.
}
*comp = c;
if (c == 0) {
// Hey look, we found it.
int32 offset = m_blockStart + (ptr - m_block);
ptr2 = XP_STRCHR(ptr2, LINEBREAK_START);
if (!ptr2) {
// We found the right place, but the rest of
// the line is off the end of the buffer.
if (offset > m_blockStart + LINEBREAK_LEN) {
m_blockStart = offset - LINEBREAK_LEN;
goto RELOAD;
}
// We couldn't find it, even though we're at the beginning
// of the buffer. This means that our buffer is too small.
goto GROWBUFFER;
}
return msg_GroupRecord::Create(parent, ptr, -1, offset);
}
}
/* NOTREACHED */
return NULL;
GROWBUFFER:
tmp = new char [m_blockSize + 512 + 1];
if (!tmp) return NULL;
delete [] m_block;
m_block = tmp;
m_blockSize += 512;
RELOAD:
if (m_blockStart + m_blockSize > m_fileSize) {
m_blockStart = m_fileSize - m_blockSize;
if (m_blockStart < m_fileStart) m_blockStart = m_fileStart;
}
OpenGroupFile();
if (!m_groupFile) return NULL;
XP_FileSeek(m_groupFile, m_blockStart, SEEK_SET);
int length = XP_FileRead(m_block, m_blockSize, m_groupFile);
if (length < 0) length = 0;
m_block[length] = '\0';
goto RESTART;
}
msg_GroupRecord*
MSG_NewsHost::LoadSingleEntry(msg_GroupRecord* parent, const char* groupname,
int32 min, int32 max)
{
OpenGroupFile();
if (!m_groupFile) return NULL;
#ifdef DEBUG
if (parent != m_groupTree) {
char* pname = parent->GetFullName();
if (pname) {
XP_ASSERT(XP_STRNCMP(pname, groupname, XP_STRLEN(pname)) == 0);
delete [] pname;
pname = NULL;
}
}
#endif
if (min < m_fileStart) min = m_fileStart;
if (max < m_fileStart || max > m_fileSize) max = m_fileSize;
msg_GroupRecord* result = NULL;
int32 comp = 1;
// First, check if we happen to already have the line for this group in
// memory.
if (m_block[0]) {
result = FindGroupInBlock(parent, groupname, &comp);
}
while (!result && comp != 0 && min < max) {
int32 mid = (min + max) / 2;
m_blockStart = mid - LINEBREAK_LEN;
XP_FileSeek(m_groupFile, m_blockStart, SEEK_SET);
int length = XP_FileRead(m_block, m_blockSize, m_groupFile);
if (length < 0) length = 0;
m_block[length] = '\0';
result = FindGroupInBlock(parent, groupname, &comp);
if (comp > 0) {
min = mid + 1;
} else {
max = mid;
}
}
return result;
}
msg_GroupRecord*
MSG_NewsHost::FindOrCreateGroup(const char* groupname,
int* statusOfMakingGroup)
{
char buf[256];
msg_GroupRecord* parent = m_groupTree;
const char* start = groupname;
XP_ASSERT(start && *start);
if (!start || !*start) return NULL;
XP_ASSERT(*start != '.'); // groupnames can't start with ".".
if (*start == '.') return NULL;
while (*start) {
if (*start == '.') start++;
const char* end = XP_STRCHR(start, '.');
if (!end) end = start + XP_STRLEN(start);
int length = end - start;
XP_ASSERT(length > 0); // groupnames can't contain ".." or end in
// a ".".
if (length <= 0) return NULL;
XP_ASSERT(length < sizeof(buf));
if (length >= sizeof(buf)) return NULL;
XP_STRNCPY_SAFE(buf, start, length + 1);
buf[length] = '\0';
msg_GroupRecord* prev = parent;
msg_GroupRecord* ptr;
int comp = 0; // Initializing to zero.
for (ptr = parent->GetChildren() ; ptr ; ptr = ptr->GetSibling()) {
comp = msg_GroupRecord::GroupNameCompare(ptr->GetPartName(), buf);
if (comp >= 0) break;
prev = ptr;
}
if (ptr == NULL || comp != 0) {
// We don't have this one in memory. See if we can load it in.
if (!m_inhaled) {
if (ptr == NULL) {
ptr = parent->GetSiblingOrAncestorSibling();
}
length = end - groupname;
char* tmp = new char[length + 1];
if (!tmp) return NULL;
XP_STRNCPY_SAFE(tmp, groupname, length + 1);
tmp[length] = '\0';
ptr = LoadSingleEntry(parent, tmp,
prev->GetFileOffset(),
ptr ? ptr->GetFileOffset() : m_fileSize);
delete [] tmp;
tmp = NULL;
} else {
ptr = NULL;
}
if (!ptr) {
m_groupTreeDirty = 2;
ptr = msg_GroupRecord::Create(parent, buf, time(0),
m_uniqueId++, 0);
if (!ptr) return NULL;
}
}
parent = ptr;
start = end;
}
int status = parent->SetIsGroup(TRUE);
m_groupTreeDirty |= status;
if (statusOfMakingGroup) *statusOfMakingGroup = status;
return parent;
}
int
MSG_NewsHost::NoticeNewGroup(const char* groupname)
{
int status = 0;
msg_GroupRecord* group = FindOrCreateGroup(groupname, &status);
if (!group) return MK_OUT_OF_MEMORY;
return status;
}
int
MSG_NewsHost::AssureAllDescendentsLoaded(msg_GroupRecord* group)
{
int status = 0;
XP_ASSERT(group);
if (!group) return -1;
if (group->IsDescendentsLoaded()) return 0;
m_blockStart = group->GetFileOffset();
XP_ASSERT(group->GetFileOffset() > 0);
if (group->GetFileOffset() == 0) return -1;
InhaleState state;
state.tree = m_groupTree;
state.position = m_blockStart;
state.onlyIfChild = group;
state.lastInhaled = NULL;
OpenGroupFile();
XP_ASSERT(m_groupFile);
if (!m_groupFile) return -1;
XP_FileSeek(m_groupFile, group->GetFileOffset(), SEEK_SET);
char* buffer = NULL;
uint32 buffer_size = 0;
uint32 buffer_fp = 0;
do {
status = XP_FileRead(m_block, m_blockSize, m_groupFile);
if (status <= 0) break;
status = msg_LineBuffer(m_block, status,
&buffer, &buffer_size, &buffer_fp,
FALSE,
#ifdef XP_OS2
(int32 (_Optlink*) (char*,uint32,void*))
#endif
MSG_NewsHost::InhaleLine, &state);
} while (status >= 0);
if (status >= 0 && buffer_fp > 0) {
status = InhaleLine(buffer, buffer_fp, &state);
}
if (status == -2) status = 0; // Special status meaning we ran out of
// lines that are children of the given
// group.
FREEIF(buffer);
m_block[0] = '\0';
if (status >= 0) group->SetIsDescendentsLoaded(TRUE);
return status;
}
void MSG_NewsHost::GroupNotFound(const char *groupName, XP_Bool opening)
{
// if no group command has succeeded, don't blow away categories.
// The server might be hanging...
if (!opening && !m_groupSucceeded)
return;
msg_GroupRecord* group = FindOrCreateGroup(groupName);
if (group && (group->IsCategory() || opening))
{
MSG_FolderInfoNews *newsInfo = FindGroup(groupName);
group->SetDoesNotExistOnServer(TRUE);
m_groupTreeDirty |= 2; // deleting a group has to force a rewrite anyway
if (group->IsCategory())
{
MSG_FolderInfoNews *catCont = GetCategoryContainerFolderInfo(groupName);
if (catCont)
{
MSG_FolderInfo *parentCategory = catCont->FindParentOf(newsInfo);
if (parentCategory)
parentCategory->RemoveSubFolder(newsInfo);
}
}
else if (newsInfo)
m_master->BroadcastFolderDeleted (newsInfo);
if (newsInfo)
{
XPPtrArray* infolist = (XPPtrArray*) m_hostinfo->GetSubFolders();
infolist->Remove(newsInfo);
m_groups->Remove(newsInfo);
m_dirty = TRUE;
}
}
}
int MSG_NewsHost::ReorderGroup (MSG_FolderInfoNews *groupToMove, MSG_FolderInfoNews *groupToMoveBefore, int32 *newIdx)
{
// Do the list maintenance to reorder a newsgroup. The news host has a list, and
// so does the FolderInfo which represents the host in the hierarchy tree
int err = MK_MSG_CANT_MOVE_FOLDER;
XPPtrArray *infoList = m_hostinfo->GetSubFolders();
if (groupToMove && groupToMoveBefore && infoList)
{
if (m_groups->Remove (groupToMove))
{
// Not necessary to remove from infoList here because the folderPane does that (urk)
// Unsubscribed groups are in the lists, but not in the view
MSG_FolderInfo *group = NULL;
int idxInView, idxInData;
int idxInHostInfo = 0;
XP_Bool foundIdxInHostInfo = FALSE;
for (idxInData = 0, idxInView = -1; idxInData < m_groups->GetSize(); idxInData++)
{
group = m_groups->GetAt (idxInData);
MSG_FolderInfo *groupInHostInfo = (MSG_FolderInfo *) infoList->GetAt(idxInHostInfo);
if (group->CanBeInFolderPane())
idxInView++;
if (group == groupToMoveBefore)
break;
if (groupInHostInfo == groupToMoveBefore)
foundIdxInHostInfo = TRUE;
else if (!foundIdxInHostInfo)
idxInHostInfo++;
}
if (idxInView != -1)
{
m_groups->InsertAt (idxInData, groupToMove); // the index has to be the same, right?
infoList->InsertAt (idxInHostInfo, groupToMove);
MarkDirty (); // Make sure the newsrc gets written out in the new order
*newIdx = idxInView; // Tell the folder pane where the new item belongs
err = 0;
}
}
}
return err;
}