//
// VMime library (http://www.vmime.org)
// Copyright (C) 2002 Vincent Richard <vincent@vmime.org>
//
// This program is free software; you can redistribute it and/or
// modify it under the terms of the GNU General Public License as
// published by the Free Software Foundation; either version 3 of
// the License, or (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
// General Public License for more details.
//
// You should have received a copy of the GNU General Public License along
// with this program; if not, write to the Free Software Foundation, Inc.,
// 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
//
// Linking this library statically or dynamically with other modules is making
// a combined work based on this library. Thus, the terms and conditions of
// the GNU General Public License cover the whole combination.
//
#include "vmime/config.hpp"
#if VMIME_HAVE_MESSAGING_FEATURES && VMIME_HAVE_MESSAGING_PROTO_MAILDIR
#include "vmime/net/maildir/maildirFolder.hpp"
#include "vmime/net/maildir/maildirStore.hpp"
#include "vmime/net/maildir/maildirMessage.hpp"
#include "vmime/net/maildir/maildirUtils.hpp"
#include "vmime/net/maildir/maildirFormat.hpp"
#include "vmime/net/maildir/maildirFolderStatus.hpp"
#include "vmime/message.hpp"
#include "vmime/exception.hpp"
#include "vmime/platform.hpp"
#include "vmime/utility/outputStreamAdapter.hpp"
#include "vmime/utility/inputStreamStringAdapter.hpp"
namespace vmime {
namespace net {
namespace maildir {
maildirFolder::maildirFolder(
const folder::path& path,
const shared_ptr <maildirStore>& store
)
: m_store(store),
m_path(path),
m_name(path.isEmpty() ? folder::path::component("") : path.getLastComponent()),
m_mode(-1),
m_open(false),
m_unreadMessageCount(0),
m_messageCount(0) {
store->registerFolder(this);
}
maildirFolder::~maildirFolder() {
try {
shared_ptr <maildirStore> store = m_store.lock();
if (store) {
if (m_open) {
close(false);
}
store->unregisterFolder(this);
} else if (m_open) {
close(false);
}
} catch (...) {
// Don't throw in destructor
}
}
void maildirFolder::onStoreDisconnected() {
m_store.reset();
}
int maildirFolder::getMode() const {
if (!isOpen()) {
throw exceptions::illegal_state("Folder not open");
}
return m_mode;
}
const folderAttributes maildirFolder::getAttributes() {
folderAttributes attribs;
if (m_path.isEmpty()) {
attribs.setType(folderAttributes::TYPE_CONTAINS_FOLDERS);
} else {
attribs.setType(folderAttributes::TYPE_CONTAINS_FOLDERS | folderAttributes::TYPE_CONTAINS_MESSAGES);
}
if (m_store.lock()->getFormat()->folderHasSubfolders(m_path)) {
attribs.setFlags(folderAttributes::FLAG_HAS_CHILDREN); // contains at least one sub-folder
}
return attribs;
}
const folder::path::component maildirFolder::getName() const {
return m_name;
}
const folder::path maildirFolder::getFullPath() const {
return m_path;
}
void maildirFolder::open(const int mode, bool /* failIfModeIsNotAvailable */) {
shared_ptr <maildirStore> store = m_store.lock();
if (!store) {
throw exceptions::illegal_state("Store disconnected");
} else if (isOpen()) {
throw exceptions::illegal_state("Folder is already open");
} else if (!exists()) {
throw exceptions::illegal_state("Folder does not exist");
}
scanFolder();
m_open = true;
m_mode = mode;
}
void maildirFolder::close(const bool expunge) {
shared_ptr <maildirStore> store = m_store.lock();
if (!store) {
throw exceptions::illegal_state("Store disconnected");
}
if (!isOpen()) {
throw exceptions::illegal_state("Folder not open");
}
if (expunge) {
this->expunge();
}
m_open = false;
m_mode = -1;
onClose();
}
void maildirFolder::onClose() {
for (std::vector <maildirMessage*>::iterator it = m_messages.begin() ;
it != m_messages.end() ; ++it) {
(*it)->onFolderClosed();
}
m_messages.clear();
}
void maildirFolder::registerMessage(maildirMessage* msg) {
m_messages.push_back(msg);
}
void maildirFolder::unregisterMessage(maildirMessage* msg) {
std::vector <maildirMessage*>::iterator it =
std::find(m_messages.begin(), m_messages.end(), msg);
if (it != m_messages.end()) {
m_messages.erase(it);
}
}
void maildirFolder::create(const folderAttributes& /* attribs */) {
shared_ptr <maildirStore> store = m_store.lock();
if (!store) {
throw exceptions::illegal_state("Store disconnected");
} else if (isOpen()) {
throw exceptions::illegal_state("Folder is open");
} else if (exists()) {
throw exceptions::illegal_state("Folder already exists");
} else if (!store->isValidFolderName(m_name)) {
throw exceptions::invalid_folder_name();
}
// Create directory on file system
try {
store->getFormat()->createFolder(m_path);
} catch (exceptions::filesystem_exception& e) {
throw exceptions::command_error("CREATE", "", "File system exception", e);
}
// Notify folder created
shared_ptr <events::folderEvent> event =
make_shared <events::folderEvent>(
dynamicCast <folder>(shared_from_this()),
events::folderEvent::TYPE_CREATED,
m_path, m_path
);
notifyFolder(event);
}
void maildirFolder::destroy() {
shared_ptr <maildirStore> store = m_store.lock();
if (!store) {
throw exceptions::illegal_state("Store disconnected");
} else if (isOpen()) {
throw exceptions::illegal_state("Folder is open");
}
// Delete folder
try {
store->getFormat()->destroyFolder(m_path);
} catch (std::exception&) {
// Ignore exception: anyway, we can't recover from this...
}
// Notify folder deleted
shared_ptr <events::folderEvent> event =
make_shared <events::folderEvent>(
dynamicCast <folder>(shared_from_this()),
events::folderEvent::TYPE_DELETED,
m_path, m_path
);
notifyFolder(event);
}
bool maildirFolder::exists() {
shared_ptr <maildirStore> store = m_store.lock();
return store->getFormat()->folderExists(m_path);
}
bool maildirFolder::isOpen() const {
return m_open;
}
void maildirFolder::scanFolder() {
shared_ptr <maildirStore> store = m_store.lock();
try {
m_messageCount = 0;
m_unreadMessageCount = 0;
shared_ptr <utility::fileSystemFactory> fsf = platform::getHandler()->getFileSystemFactory();
utility::file::path newDirPath =
store->getFormat()->folderPathToFileSystemPath(m_path, maildirFormat::NEW_DIRECTORY);
shared_ptr <utility::file> newDir = fsf->create(newDirPath);
utility::file::path curDirPath =
store->getFormat()->folderPathToFileSystemPath(m_path, maildirFormat::CUR_DIRECTORY);
shared_ptr <utility::file> curDir = fsf->create(curDirPath);
// New received messages (new/)
shared_ptr <utility::fileIterator> nit = newDir->getFiles();
std::vector <utility::file::path::component> newMessageFilenames;
while (nit->hasMoreElements()) {
shared_ptr <utility::file> file = nit->nextElement();
if (maildirUtils::isMessageFile(*file)) {
newMessageFilenames.push_back(file->getFullPath().getLastComponent());
}
}
// Current messages (cur/)
shared_ptr <utility::fileIterator> cit = curDir->getFiles();
std::vector <utility::file::path::component> curMessageFilenames;
while (cit->hasMoreElements()) {
shared_ptr <utility::file> file = cit->nextElement();
if (maildirUtils::isMessageFile(*file)) {
curMessageFilenames.push_back(file->getFullPath().getLastComponent());
}
}
// Update/delete existing messages (found in previous scan)
for (size_t i = 0 ; i < m_messageInfos.size() ; ++i) {
messageInfos& msgInfos = m_messageInfos[i];
// NOTE: the flags may have changed (eg. moving from 'new' to 'cur'
// may imply the 'S' flag) and so the filename. That's why we use
// "maildirUtils::messageIdComparator" to compare only the 'unique'
// portion of the filename...
if (msgInfos.type == messageInfos::TYPE_CUR) {
const std::vector <utility::file::path::component>::iterator pos =
std::find_if(
curMessageFilenames.begin(),
curMessageFilenames.end(),
maildirUtils::messageIdComparator(msgInfos.path)
);
// If we cannot find this message in the 'cur' directory,
// it means it has been deleted (and expunged).
if (pos == curMessageFilenames.end()) {
msgInfos.type = messageInfos::TYPE_DELETED;
// Otherwise, update its information.
} else {
msgInfos.path = *pos;
curMessageFilenames.erase(pos);
}
}
}
m_messageInfos.reserve(m_messageInfos.size()
+ newMessageFilenames.size() + curMessageFilenames.size());
// Add new messages from 'new': we are responsible to move the files
// from the 'new' directory to the 'cur' directory, and append them
// to our message list.
for (std::vector <utility::file::path::component>::const_iterator
it = newMessageFilenames.begin() ; it != newMessageFilenames.end() ; ++it) {
const utility::file::path::component newFilename =
maildirUtils::buildFilename(maildirUtils::extractId(*it), 0);
// Move messages from 'new' to 'cur'
shared_ptr <utility::file> file = fsf->create(newDirPath / *it);
file->rename(curDirPath / newFilename);
// Append to message list
messageInfos msgInfos;
msgInfos.path = newFilename;
if (maildirUtils::extractFlags(msgInfos.path) & message::FLAG_DELETED) {
msgInfos.type = messageInfos::TYPE_DELETED;
} else {
msgInfos.type = messageInfos::TYPE_CUR;
}
m_messageInfos.push_back(msgInfos);
}
// Add new messages from 'cur': the files have already been moved
// from 'new' to 'cur'. Just append them to our message list.
for (std::vector <utility::file::path::component>::const_iterator
it = curMessageFilenames.begin() ; it != curMessageFilenames.end() ; ++it) {
// Append to message list
messageInfos msgInfos;
msgInfos.path = *it;
if (maildirUtils::extractFlags(msgInfos.path) & message::FLAG_DELETED) {
msgInfos.type = messageInfos::TYPE_DELETED;
} else {
msgInfos.type = messageInfos::TYPE_CUR;
}
m_messageInfos.push_back(msgInfos);
}
// Update message count
size_t unreadMessageCount = 0;
for (std::vector <messageInfos>::const_iterator
it = m_messageInfos.begin() ; it != m_messageInfos.end() ; ++it) {
if ((maildirUtils::extractFlags((*it).path) & message::FLAG_SEEN) == 0) {
++unreadMessageCount;
}
}
m_unreadMessageCount = unreadMessageCount;
m_messageCount = static_cast <size_t>(m_messageInfos.size());
} catch (exceptions::filesystem_exception&) {
// Should not happen...
}
}
shared_ptr <message> maildirFolder::getMessage(const size_t num) {
if (!isOpen()) {
throw exceptions::illegal_state("Folder not open");
}
if (num < 1 || num > m_messageCount) {
throw exceptions::message_not_found();
}
return make_shared <maildirMessage>(dynamicCast <maildirFolder>(shared_from_this()), num);
}
std::vector <shared_ptr <message> > maildirFolder::getMessages(const messageSet& msgs) {
if (!isOpen()) {
throw exceptions::illegal_state("Folder not open");
}
if (msgs.isNumberSet()) {
const std::vector <size_t> numbers = maildirUtils::messageSetToNumberList(msgs, m_messageCount);
std::vector <shared_ptr <message> > messages;
shared_ptr <maildirFolder> thisFolder = dynamicCast <maildirFolder>(shared_from_this());
for (std::vector <size_t>::const_iterator it = numbers.begin() ; it != numbers.end() ; ++it) {
if (*it < 1|| *it > m_messageCount) {
throw exceptions::message_not_found();
}
messages.push_back(make_shared <maildirMessage>(thisFolder, *it));
}
return messages;
} else {
throw exceptions::operation_not_supported();
}
}
size_t maildirFolder::getMessageCount() {
return m_messageCount;
}
shared_ptr <folder> maildirFolder::getFolder(const folder::path::component& name) {
shared_ptr <maildirStore> store = m_store.lock();
if (!store) {
throw exceptions::illegal_state("Store disconnected");
}
return shared_ptr <maildirFolder>(new maildirFolder(m_path / name, store));
}
std::vector <shared_ptr <folder> > maildirFolder::getFolders(const bool recursive) {
shared_ptr <maildirStore> store = m_store.lock();
if (!isOpen() && !store) {
throw exceptions::illegal_state("Store disconnected");
}
std::vector <shared_ptr <folder> > list;
listFolders(list, recursive);
return list;
}
void maildirFolder::listFolders(std::vector <shared_ptr <folder> >& list, const bool recursive) {
shared_ptr <maildirStore> store = m_store.lock();
try {
std::vector <folder::path> pathList =
store->getFormat()->listFolders(m_path, recursive);
list.reserve(pathList.size());
for (std::vector <folder::path>::size_type i = 0, n = pathList.size() ; i < n ; ++i) {
shared_ptr <maildirFolder> subFolder(
new maildirFolder(pathList[i], store)
);
list.push_back(subFolder);
}
} catch (exceptions::filesystem_exception& e) {
throw exceptions::command_error("LIST", "", "", e);
}
}
void maildirFolder::rename(const folder::path& newPath) {
shared_ptr <maildirStore> store = m_store.lock();
if (!store) {
throw exceptions::illegal_state("Store disconnected");
} else if (m_path.isEmpty() || newPath.isEmpty()) {
throw exceptions::illegal_operation("Cannot rename root folder");
} else if (!store->isValidFolderName(newPath.getLastComponent())) {
throw exceptions::invalid_folder_name();
}
// Rename the directory on the file system
try {
store->getFormat()->renameFolder(m_path, newPath);
} catch (vmime::exception& e) {
throw exceptions::command_error("RENAME", "", "", e);
}
// Notify folder renamed
folder::path oldPath(m_path);
m_path = newPath;
m_name = newPath.getLastComponent();
shared_ptr <events::folderEvent> event =
make_shared <events::folderEvent>(
dynamicCast <folder>(shared_from_this()),
events::folderEvent::TYPE_RENAMED,
oldPath, newPath
);
notifyFolder(event);
// Notify folders with the same path
for (std::list <maildirFolder*>::iterator it = store->m_folders.begin() ;
it != store->m_folders.end() ; ++it) {
if ((*it) != this && (*it)->getFullPath() == oldPath) {
(*it)->m_path = newPath;
(*it)->m_name = newPath.getLastComponent();
shared_ptr <events::folderEvent> event =
make_shared <events::folderEvent>(
dynamicCast <folder>((*it)->shared_from_this()),
events::folderEvent::TYPE_RENAMED,
oldPath, newPath
);
(*it)->notifyFolder(event);
} else if ((*it) != this && oldPath.isParentOf((*it)->getFullPath())) {
folder::path oldPath((*it)->m_path);
(*it)->m_path.renameParent(oldPath, newPath);
shared_ptr <events::folderEvent> event =
make_shared <events::folderEvent>(
dynamicCast <folder>((*it)->shared_from_this()),
events::folderEvent::TYPE_RENAMED,
oldPath, (*it)->m_path
);
(*it)->notifyFolder(event);
}
}
}
void maildirFolder::deleteMessages(const messageSet& msgs) {
// Mark messages as deleted
setMessageFlags(msgs, message::FLAG_DELETED, message::FLAG_MODE_ADD);
}
void maildirFolder::setMessageFlags(
const messageSet& msgs,
const int flags,
const int mode
) {
shared_ptr <maildirStore> store = m_store.lock();
if (!store) {
throw exceptions::illegal_state("Store disconnected");
} else if (!isOpen()) {
throw exceptions::illegal_state("Folder not open");
} else if (m_mode == MODE_READ_ONLY) {
throw exceptions::illegal_state("Folder is read-only");
}
if (msgs.isNumberSet()) {
const std::vector <size_t> nums = maildirUtils::messageSetToNumberList(msgs, m_messageCount);
// Change message flags
shared_ptr <utility::fileSystemFactory> fsf = platform::getHandler()->getFileSystemFactory();
utility::file::path curDirPath = store->getFormat()->
folderPathToFileSystemPath(m_path, maildirFormat::CUR_DIRECTORY);
for (std::vector <size_t>::const_iterator it =
nums.begin() ; it != nums.end() ; ++it) {
const size_t num = *it - 1;
try {
const utility::file::path::component path = m_messageInfos[num].path;
shared_ptr <utility::file> file = fsf->create(curDirPath / path);
int newFlags = maildirUtils::extractFlags(path);
switch (mode) {
case message::FLAG_MODE_ADD: newFlags |= flags; break;
case message::FLAG_MODE_REMOVE: newFlags &= ~flags; break;
default:
case message::FLAG_MODE_SET: newFlags = flags; break;
}
const utility::file::path::component newPath =
maildirUtils::buildFilename(maildirUtils::extractId(path), newFlags);
file->rename(curDirPath / newPath);
if (flags & message::FLAG_DELETED) {
m_messageInfos[num].type = messageInfos::TYPE_DELETED;
} else {
m_messageInfos[num].type = messageInfos::TYPE_CUR;
}
m_messageInfos[num].path = newPath;
} catch (exceptions::filesystem_exception& e) {
// Ignore (not important)
}
}
// Update local flags
switch (mode) {
case message::FLAG_MODE_ADD: {
for (std::vector <maildirMessage*>::iterator it =
m_messages.begin() ; it != m_messages.end() ; ++it) {
if (std::binary_search(nums.begin(), nums.end(), (*it)->getNumber()) &&
(*it)->m_flags != maildirMessage::FLAG_UNDEFINED) {
(*it)->m_flags |= flags;
}
}
break;
}
case message::FLAG_MODE_REMOVE: {
for (std::vector <maildirMessage*>::iterator it =
m_messages.begin() ; it != m_messages.end() ; ++it) {
if (std::binary_search(nums.begin(), nums.end(), (*it)->getNumber()) &&
(*it)->m_flags != maildirMessage::FLAG_UNDEFINED) {
(*it)->m_flags &= ~flags;
}
}
break;
}
default:
case message::FLAG_MODE_SET: {
for (std::vector <maildirMessage*>::iterator it =
m_messages.begin() ; it != m_messages.end() ; ++it) {
if (std::binary_search(nums.begin(), nums.end(), (*it)->getNumber()) &&
(*it)->m_flags != maildirMessage::FLAG_UNDEFINED) {
(*it)->m_flags = flags;
}
}
break;
}
}
// Notify message flags changed
shared_ptr <events::messageChangedEvent> event =
make_shared <events::messageChangedEvent>(
dynamicCast <folder>(shared_from_this()),
events::messageChangedEvent::TYPE_FLAGS,
nums
);
notifyMessageChanged(event);
// TODO: notify other folders with the same path
} else {
throw exceptions::operation_not_supported();
}
}
messageSet maildirFolder::addMessage(
const shared_ptr <vmime::message>& msg,
const int flags,
vmime::datetime* date,
utility::progressListener* progress
) {
std::ostringstream oss;
utility::outputStreamAdapter ossAdapter(oss);
msg->generate(ossAdapter);
const string& str = oss.str();
utility::inputStreamStringAdapter strAdapter(str);
return addMessage(strAdapter, str.length(), flags, date, progress);
}
messageSet maildirFolder::addMessage(
utility::inputStream& is,
const size_t size,
const int flags,
vmime::datetime* /* date */,
utility::progressListener* progress
) {
shared_ptr <maildirStore> store = m_store.lock();
if (!store) {
throw exceptions::illegal_state("Store disconnected");
} else if (!isOpen()) {
throw exceptions::illegal_state("Folder not open");
} else if (m_mode == MODE_READ_ONLY) {
throw exceptions::illegal_state("Folder is read-only");
}
shared_ptr <utility::fileSystemFactory> fsf = platform::getHandler()->getFileSystemFactory();
utility::file::path tmpDirPath = store->getFormat()->
folderPathToFileSystemPath(m_path,maildirFormat::TMP_DIRECTORY);
utility::file::path dstDirPath = store->getFormat()->
folderPathToFileSystemPath(
m_path,
flags == message::FLAG_RECENT ?
maildirFormat::NEW_DIRECTORY :
maildirFormat::CUR_DIRECTORY
);
const utility::file::path::component filename =
maildirUtils::buildFilename(maildirUtils::generateId(), ((flags == -1) ? 0 : flags));
try {
shared_ptr <utility::file> tmpDir = fsf->create(tmpDirPath);
tmpDir->createDirectory(true);
} catch (exceptions::filesystem_exception&) {
// Don't throw now, it will fail later...
}
try {
shared_ptr <utility::file> curDir = fsf->create(dstDirPath);
curDir->createDirectory(true);
} catch (exceptions::filesystem_exception&) {
// Don't throw now, it will fail later...
}
// Actually add the message
copyMessageImpl(tmpDirPath, dstDirPath, filename, is, size, progress);
// Append the message to the cache list
messageInfos msgInfos;
msgInfos.path = filename;
msgInfos.type = messageInfos::TYPE_CUR;
m_messageInfos.push_back(msgInfos);
m_messageCount++;
if ((flags == -1) || !(flags & message::FLAG_SEEN)) {
m_unreadMessageCount++;
}
// Notification
std::vector <size_t> nums;
nums.push_back(m_messageCount);
shared_ptr <events::messageCountEvent> event =
make_shared <events::messageCountEvent>(
dynamicCast <folder>(shared_from_this()),
events::messageCountEvent::TYPE_ADDED,
nums
);
notifyMessageCount(event);
// Notify folders with the same path
for (std::list <maildirFolder*>::iterator it = store->m_folders.begin() ;
it != store->m_folders.end() ; ++it) {
if ((*it) != this && (*it)->getFullPath() == m_path) {
(*it)->m_messageCount = m_messageCount;
(*it)->m_unreadMessageCount = m_unreadMessageCount;
(*it)->m_messageInfos.resize(m_messageInfos.size());
std::copy(m_messageInfos.begin(), m_messageInfos.end(), (*it)->m_messageInfos.begin());
shared_ptr <events::messageCountEvent> event =
make_shared <events::messageCountEvent>(
dynamicCast <folder>((*it)->shared_from_this()),
events::messageCountEvent::TYPE_ADDED,
nums
);
(*it)->notifyMessageCount(event);
}
}
return messageSet::empty();
}
void maildirFolder::copyMessageImpl(
const utility::file::path& tmpDirPath,
const utility::file::path& dstDirPath,
const utility::file::path::component& filename,
utility::inputStream& is, const size_t size,
utility::progressListener* progress
) {
shared_ptr <utility::fileSystemFactory> fsf = platform::getHandler()->getFileSystemFactory();
shared_ptr <utility::file> file = fsf->create(tmpDirPath / filename);
if (progress) {
progress->start(size);
}
// First, write the message into 'tmp'...
try {
file->createFile();
shared_ptr <utility::fileWriter> fw = file->getFileWriter();
shared_ptr <utility::outputStream> os = fw->getOutputStream();
byte_t buffer[65536];
size_t total = 0;
while (!is.eof()) {
const size_t read = is.read(buffer, sizeof(buffer));
if (read != 0) {
os->write(buffer, read);
total += read;
}
if (progress) {
progress->progress(total, size);
}
}
os->flush();
} catch (exception& e) {
if (progress) {
progress->stop(size);
}
// Delete temporary file
try {
shared_ptr <utility::file> file = fsf->create(tmpDirPath / filename);
file->remove();
} catch (exceptions::filesystem_exception&) {
// Ignore
}
throw exceptions::command_error("ADD", "", "", e);
}
// ...then, move it to 'cur'
try {
file->rename(dstDirPath / filename);
} catch (exception& e) {
if (progress) {
progress->stop(size);
}
// Delete temporary file
try {
file->remove();
shared_ptr <utility::file> file = fsf->create(dstDirPath / filename);
file->remove();
} catch (exceptions::filesystem_exception&) {
// Ignore
}
throw exceptions::command_error("ADD", "", "", e);
}
if (progress) {
progress->stop(size);
}
}
messageSet maildirFolder::copyMessages(const folder::path& dest, const messageSet& msgs) {
shared_ptr <maildirStore> store = m_store.lock();
if (!store) {
throw exceptions::illegal_state("Store disconnected");
} else if (!isOpen()) {
throw exceptions::illegal_state("Folder not open");
}
shared_ptr <utility::fileSystemFactory> fsf = platform::getHandler()->getFileSystemFactory();
utility::file::path curDirPath =
store->getFormat()->folderPathToFileSystemPath(m_path, maildirFormat::CUR_DIRECTORY);
utility::file::path destCurDirPath = store->getFormat()->
folderPathToFileSystemPath(dest, maildirFormat::CUR_DIRECTORY);
utility::file::path destTmpDirPath = store->getFormat()->
folderPathToFileSystemPath(dest, maildirFormat::TMP_DIRECTORY);
// Create destination directories
try {
shared_ptr <utility::file> destTmpDir = fsf->create(destTmpDirPath);
destTmpDir->createDirectory(true);
} catch (exceptions::filesystem_exception&) {
// Don't throw now, it will fail later...
}
try {
shared_ptr <utility::file> destCurDir = fsf->create(destCurDirPath);
destCurDir->createDirectory(true);
} catch (exceptions::filesystem_exception&) {
// Don't throw now, it will fail later...
}
// Copy messages
const std::vector <size_t> nums = maildirUtils::messageSetToNumberList(msgs, m_messageCount);
try {
for (std::vector <size_t>::const_iterator it =
nums.begin() ; it != nums.end() ; ++it) {
const size_t num = *it;
const messageInfos& msg = m_messageInfos[num - 1];
const int flags = maildirUtils::extractFlags(msg.path);
const utility::file::path::component filename =
maildirUtils::buildFilename(maildirUtils::generateId(), flags);
shared_ptr <utility::file> file = fsf->create(curDirPath / msg.path);
shared_ptr <utility::fileReader> fr = file->getFileReader();
shared_ptr <utility::inputStream> is = fr->getInputStream();
copyMessageImpl(destTmpDirPath, destCurDirPath,
filename, *is, file->getLength(), NULL);
}
} catch (exception& e) {
notifyMessagesCopied(dest);
throw exceptions::command_error("COPY", "", "", e);
}
notifyMessagesCopied(dest);
return messageSet::empty();
}
void maildirFolder::notifyMessagesCopied(const folder::path& dest) {
shared_ptr <maildirStore> store = m_store.lock();
for (std::list <maildirFolder*>::iterator it = store->m_folders.begin() ;
it != store->m_folders.end() ; ++it) {
if ((*it) != this && (*it)->getFullPath() == dest) {
// We only need to update the first folder we found as calling
// status() will notify all the folders with the same path.
size_t count, unseen;
(*it)->status(count, unseen);
return;
}
}
}
void maildirFolder::status(size_t& count, size_t& unseen) {
count = 0;
unseen = 0;
shared_ptr <folderStatus> status = getStatus();
count = status->getMessageCount();
unseen = status->getUnseenCount();
m_messageCount = count;
}
shared_ptr <folderStatus> maildirFolder::getStatus() {
shared_ptr <maildirStore> store = m_store.lock();
const size_t oldCount = m_messageCount;
scanFolder();
shared_ptr <maildirFolderStatus> status = make_shared <maildirFolderStatus>();
status->setMessageCount(m_messageCount);
status->setUnseenCount(m_unreadMessageCount);
// Notify message count changed (new messages)
if (m_messageCount > oldCount) {
std::vector <size_t> nums;
nums.reserve(m_messageCount - oldCount);
for (size_t i = oldCount + 1, j = 0 ; i <= m_messageCount ; ++i, ++j) {
nums[j] = i;
}
shared_ptr <events::messageCountEvent> event =
make_shared <events::messageCountEvent>(
dynamicCast <folder>(shared_from_this()),
events::messageCountEvent::TYPE_ADDED,
nums
);
notifyMessageCount(event);
// Notify folders with the same path
for (std::list <maildirFolder*>::iterator it = store->m_folders.begin() ;
it != store->m_folders.end() ; ++it) {
if ((*it) != this && (*it)->getFullPath() == m_path) {
(*it)->m_messageCount = m_messageCount;
(*it)->m_unreadMessageCount = m_unreadMessageCount;
(*it)->m_messageInfos.resize(m_messageInfos.size());
std::copy(m_messageInfos.begin(), m_messageInfos.end(), (*it)->m_messageInfos.begin());
shared_ptr <events::messageCountEvent> event =
make_shared <events::messageCountEvent>(
dynamicCast <folder>((*it)->shared_from_this()),
events::messageCountEvent::TYPE_ADDED,
nums
);
(*it)->notifyMessageCount(event);
}
}
}
return status;
}
void maildirFolder::expunge() {
shared_ptr <maildirStore> store = m_store.lock();
if (!store) {
throw exceptions::illegal_state("Store disconnected");
} else if (!isOpen()) {
throw exceptions::illegal_state("Folder not open");
} else if (m_mode == MODE_READ_ONLY) {
throw exceptions::illegal_state("Folder is read-only");
}
shared_ptr <utility::fileSystemFactory> fsf = platform::getHandler()->getFileSystemFactory();
utility::file::path curDirPath = store->getFormat()->
folderPathToFileSystemPath(m_path, maildirFormat::CUR_DIRECTORY);
std::vector <size_t> nums;
size_t unreadCount = 0;
for (size_t num = 1 ; num <= m_messageCount ; ++num) {
messageInfos& infos = m_messageInfos[num - 1];
if (infos.type == messageInfos::TYPE_DELETED) {
nums.push_back(num);
for (std::vector <maildirMessage*>::iterator it =
m_messages.begin() ; it != m_messages.end() ; ++it) {
if ((*it)->m_num == num) {
(*it)->m_expunged = true;
} else if ((*it)->m_num > num) {
(*it)->m_num--;
}
}
if (maildirUtils::extractFlags(infos.path) & message::FLAG_SEEN) {
++unreadCount;
}
// Delete file from file system
try {
shared_ptr <utility::file> file = fsf->create(curDirPath / infos.path);
file->remove();
} catch (exceptions::filesystem_exception& e) {
// Ignore (not important)
}
}
}
if (!nums.empty()) {
for (std::vector <size_t>::size_type i = nums.size() ; i != 0 ; --i) {
m_messageInfos.erase(m_messageInfos.begin() + (i - 1));
}
}
m_messageCount -= static_cast <size_t>(nums.size());
m_unreadMessageCount -= unreadCount;
// Notify message expunged
shared_ptr <events::messageCountEvent> event =
make_shared <events::messageCountEvent>(
dynamicCast <folder>(shared_from_this()),
events::messageCountEvent::TYPE_REMOVED,
nums
);
notifyMessageCount(event);
// Notify folders with the same path
for (std::list <maildirFolder*>::iterator it = store->m_folders.begin() ;
it != store->m_folders.end() ; ++it) {
if ((*it) != this && (*it)->getFullPath() == m_path) {
(*it)->m_messageCount = m_messageCount;
(*it)->m_unreadMessageCount = m_unreadMessageCount;
(*it)->m_messageInfos.resize(m_messageInfos.size());
std::copy(m_messageInfos.begin(), m_messageInfos.end(), (*it)->m_messageInfos.begin());
shared_ptr <events::messageCountEvent> event =
make_shared <events::messageCountEvent>(
dynamicCast <folder>((*it)->shared_from_this()),
events::messageCountEvent::TYPE_REMOVED,
nums
);
(*it)->notifyMessageCount(event);
}
}
}
shared_ptr <folder> maildirFolder::getParent() {
if (m_path.isEmpty()) {
return null;
} else {
return shared_ptr <maildirFolder>(new maildirFolder(m_path.getParent(), m_store.lock()));
}
}
shared_ptr <const store> maildirFolder::getStore() const {
return m_store.lock();
}
shared_ptr <store> maildirFolder::getStore() {
return m_store.lock();
}
void maildirFolder::fetchMessages(
std::vector <shared_ptr <message> >& msg,
const fetchAttributes& options,
utility::progressListener* progress
) {
shared_ptr <maildirStore> store = m_store.lock();
if (!store) {
throw exceptions::illegal_state("Store disconnected");
} else if (!isOpen()) {
throw exceptions::illegal_state("Folder not open");
}
if (msg.empty()) {
return;
}
const size_t total = msg.size();
size_t current = 0;
if (progress) {
progress->start(total);
}
shared_ptr <maildirFolder> thisFolder = dynamicCast <maildirFolder>(shared_from_this());
for (std::vector <shared_ptr <message> >::iterator it = msg.begin() ;
it != msg.end() ; ++it) {
dynamicCast <maildirMessage>(*it)->fetch(thisFolder, options);
if (progress) {
progress->progress(++current, total);
}
}
if (progress) {
progress->stop(total);
}
}
void maildirFolder::fetchMessage(
const shared_ptr <message>& msg,
const fetchAttributes& options
) {
shared_ptr <maildirStore> store = m_store.lock();
if (!store) {
throw exceptions::illegal_state("Store disconnected");
} else if (!isOpen()) {
throw exceptions::illegal_state("Folder not open");
}
dynamicCast <maildirMessage>(msg)->fetch(
dynamicCast <maildirFolder>(shared_from_this()),
options
);
}
std::vector <shared_ptr <message> > maildirFolder::getAndFetchMessages(
const messageSet& msgs,
const fetchAttributes& attribs
) {
if (msgs.isEmpty()) {
return std::vector <shared_ptr <message> >();
}
std::vector <shared_ptr <message> > messages = getMessages(msgs);
fetchMessages(messages, attribs);
return messages;
}
int maildirFolder::getFetchCapabilities() const {
return fetchAttributes::ENVELOPE |
fetchAttributes::STRUCTURE |
fetchAttributes::CONTENT_INFO |
fetchAttributes::FLAGS |
fetchAttributes::SIZE |
fetchAttributes::FULL_HEADER |
fetchAttributes::UID |
fetchAttributes::IMPORTANCE;
}
const utility::file::path maildirFolder::getMessageFSPath(const size_t number) const {
utility::file::path curDirPath = m_store.lock()->getFormat()->
folderPathToFileSystemPath(m_path, maildirFormat::CUR_DIRECTORY);
return curDirPath / m_messageInfos[number - 1].path;
}
std::vector <size_t> maildirFolder::getMessageNumbersStartingOnUID(const message::uid& /* uid */) {
throw exceptions::operation_not_supported();
}
} // maildir
} // net
} // vmime
#endif // VMIME_HAVE_MESSAGING_FEATURES && VMIME_HAVE_MESSAGING_PROTO_MAILDIR